5

我以前只使用反射来做一些事情,比如动态获取类并在其中设置字段值。我的谷歌搜索显示我也可以使用反射进行动态类型转换?

我的代码如下:

import entity.Shipvia;
import entity.Route;
import java.lang.reflect.Field;
import java.util.List;
import javax.persistence.EntityManager;
import javax.persistence.Persistence;
import javax.persistence.Query;

public class RetrieveResultList {

    public static List retrieveResultList(String tablename) {
        EntityManager entityManager = Persistence.createEntityManagerFactory("EntityLibraryPU").createEntityManager();
        Query query = entityManager.createNamedQuery(tablename + ".findAll");
        List<Shipvia> resultList = query.getResultList();
        return resultList;
    }
}

我正在使用此方法从数据库表中动态检索结果。因为表名总是不同的,所以我不能有 List 因为每个表都会不同。

我将如何将我传入的表名字符串转换为列表的类型?

4

1 回答 1

5

你不能这样做,即使你可以,它也是无用的,因为在编译 Java 代码时,所有泛型信息都会从 Java 代码中删除,只有强制转换存在,并且因为你将使用反射,所以不会进行强制转换.

您将能够做的最接近的事情是,而不是发送一个字符串发送一个对象。调用者代码必须说明它想要哪个类(调用者可能知道它正在使用哪种对象),并且您将使用它来使列表具有正确的泛型。

一个非常简单的实现是这样的:

List<Shipvia> shipvias = RetrieveResultList.retrieveResultList( Shipvia.class );

实现可能是这样的:

public class RetrieveResultList {

    private static final EntityManagerFactory FACTORY = Persistence.createEntityManagerFactory("EntityLibraryPU");

    public static <T> List<T> retrieveResultList(Class<T> type) {
        EntityManager entityManager = FACTORY.createEntityManager();
        String tablename = type.getName(); // figure out table name from **type**
        Query query = entityManager.createNamedQuery(tablename + ".findAll");
        List<T> resultList = query.getResultList();
        entityManager.close();
        return resultList;
    }
}

然后你应该有你正在寻找的东西。

另外,不要在每次调用此方法时创建实体管理器工厂,实体管理器工厂必须是您项目中的单例,因为它是一个非常昂贵的创建对象。您还应该在离开该方法之前关闭您创建的EntityManager 。

于 2012-05-28T15:43:23.007 回答