Java 泛型 : how to get a generic type from a method?

标签 java class generics types

虽然我的问题是关于 Java 泛型的,但我已经放了一些与 JPA 相关的代码来向您展示真实的上下文。

我正在使用 JPA 2.0 和基于 Criteria API 的查询。我所有的查询都遵循相同的模式(比较简单的属性;不需要路径导航),所以我试图编写一个通用类来处理 JPA,同时将业务逻辑保留在单独的类中。我的目标是有一个方法,给定一个实体类型和一个存储定义标准的对(字段名称 ->期望值)的映射,返回一个具有某些实体字段值的bean(或bean集合) .

我所有的实体都实现了 Persistible接口(interface),而我所有的传输对象都继承自 QueryBean .我认为那些类与问题无关,因此我跳过了它们的代码。

以下代码是我的第一种方法的片段(请假设 cb 是一个有效的 CriteriaBuilder 实例):

protected <T extends QueryBean, TT extends Persistible> Collection<T> executeQueryEntity
        (Class<T> type, Class<TT> rootType, QueryConfig queryConfig, Map<String, Object> parameters) {
    // (...) Initialization goes here

    CriteriaQuery<T> c = cb.createQuery(type);

    // FROM CLAUSE
    Root<TT> root = c.from(rootType);

    // SELECT CLAUSE
    List<String> constructorParams = queryConfig.getTargetAttributes();
    Selection<?>[] selectArray = new Selection<?>[constructorParams.size()];
    for (int i = 0; i < constructorParams.size(); i++) {
        selectArray[i] = root.get(constructorParams.get(i));
    }
    c.select(cb.construct(type, selectArray));

    // WHERE CLAUSE
    for (String filter : parameters.keySet()) {
        if (queryConfig.getFieldConfiguration().get(filter).compareUsingEquals()) {
            // SOME PREDICATE
        }
        else {
            // SOME OTHER PREDICATE
        }
    }

    // (...) Rest of the code goes here
}

我的QueryConfig界面如下:

public interface QueryConfig {

List<String> getTargetAttributes();
Map<String, FieldConfiguration> getFieldConfiguration();

}

因为我已经在使用 QueryConfig类提供有关查询的信息(例如 QueryBean 构造函数所需的参数或有关实体字段的信息),我认为从该类获取实体类型而不是将其作为类参数传递会更好。我从 this 推断质疑它不能直接完成,所以我尝试了以下解决方法:

像这样向 QueryConfig 添加一个方法:

Class< ? extends Persistible> getTargetEntity();

添加这样的中间方法:

public <T extends QueryBean> Collection<T> queryMany(Class<T> type, QueryConfig config, Map<String, Object> parameters) {
    executeQueryEntity(type, config.getTargetEntity(), parameters);
}

但是它不会编译。我相信这里解释了原因:Type mismatch for Class Generics但我实际上不明白答案。 我的问题是:有没有办法避免通过 Class< TT >执行方法的参数?这是处理问题的好方法还是我应该重构整个事情? 也欢迎对代码进行任何改进!

最佳答案

我不太明白为什么您认为您的方法会奏效: QueryConfigClass< ? extends Persistible> getTargetEntity();没有提供编译器可以引用的类型信息,为什么你认为编译器可以“猜测”你要返回的类型,并为你做类型检查?

一种方法是在 QueryConfig 中提供类型信息

public interface QueryConfig <T extends Persistable> {
  Class<T> getTargetEntity();
  List<String> getTargetAttributes();
  Map<String, FieldConfiguration> getFieldConfiguration();
}

你的 queryMany 方法可以是这样的:

public <T extends QueryBean, TT extends Persistible> 
Collection<T> queryMany(Class<T> type, 
                        QueryConfig<TT> config, 
                        Map<String, Object> parameters) {
    executeQueryEntity(type, config.getTargetEntity(), parameters);
}

关于Java 泛型 : how to get a generic type from a method?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16183105/

相关文章:

android-studio - 类型参数不在其范围内 预期 : Any Found: String

c# - 在 XML 代码注释中引用泛型类型

java - 基础Java - 如何使用方法等

c++ - 默认构造函数,为什么我的类似乎有三个?编译器何时将类视为结构?

java - 所有 Java 证书的映射在哪里?

java - 从给定纬度和经度的固定距离生成随机地理点

java - 非静态变量 this 不能从静态内容 Java 中引用

kotlin - 如何递归地实现对 Iterable 的深度扁平化?

java - API key 配置android

java - 我们如何使用java/struts2中的itext库在pdf中添加 '.svg'格式的图像