gpt4 book ai didi

java - 如何使用反射解析泛型返回类型的实际类型?

转载 作者:塔克拉玛干 更新时间:2023-11-03 04:04:01 24 4
gpt4 key购买 nike

我有一个接口(interface),它带有一个具有通用返回类型的方法,并且在运行时有一些间接实现该接口(interface)的类实例。现在我想找出每个使用反射的实现的实际返回类型

(我的想法是利用这种机制,使用接口(interface)来定义一个策略,并在运行时从一组策略实现中找到一个匹配的策略(特定的返回类型),而不必引入冗余的辅助方法公开类型)。

更具体地说,让我们考虑以下场景:

private interface DAO <I extends Serializable, E> {

public E getById (I id);
}

private abstract class AbstractDAO <T> implements DAO<Integer, T> {

@Override
public T getById (Integer id) {
// dummy implementation, just for this example
return null;
}
}

private class PersonDAO extends AbstractDAO<Person> {
}

private class PersonDAOExtension extends PersonDAO {
}

在运行时,我想找出给定类 (PersonDAOExtension.class) 将为 getById(..) 方法返回哪种类型(预期是:Person.class)。

使用反射,我可以找出从该方法返回的通用 Type。在这种情况下,它是一个 TypeVariable(但也可以是一个 Class,如果层次结构中的任何类将指定协变返回类型):

Method method = PersonDAOExtension.class.getMethod("getById", Integer.class);
Type genericReturnType = method.getGenericReturnType();
if (genericReturnType instanceof TypeVariable<?>) {
TypeVariable<?> typeVariable = (TypeVariable<?>) genericReturnType;
typeVariable.getName(); //results in "T"
}

我假设解析实际类型意味着递归到父类(super class)和接口(interface)中并转换原始(parameterizedType.getRawType())和实际 (parameterizedType.getActualTypeArguments()) 任何参数化类型的类型参数,直到找到所需的类型名称。

有没有人以前做过这个,也许一些代码片段可以帮助我实现这个?非常感谢:)


提示:我能够在运行时使用反射提取以下信息,因此保留了原始和实际类型信息:

private abstract interface DAO<I, E>
private abstract class AbstractDAO<T> extends Object implements DAO<Integer, T> [raw type:DAO<I, E>]
private class PersonDAO extends AbstractDAO<Person> [raw type:AbstractDAO<T>]
private class PersonDAOExtension extends PersonDAO

最佳答案

我能够使用 Google Guava 的 TypeToken 在一行中确定方法的通用返回类型类:

TypeToken.of(PersonDAOExtension.class)
.resolveType(PersonDAOExtension.class.getMethod("getById", Integer.class).getGenericReturnType())
.getRawType()

或者,如果您想获取类的泛型类型(就像您在接受的答案中所做的那样),而不是方法的返回类型,您可以执行以下操作:

TypeToken.of(PersonDAOExtension.class)
.resolveType(AbstractDAO.class.getTypeParameters()[0])
.getRawType()

这两种解决方案都按预期返回 Person.class

从您对已接受答案的评论来看,您似乎只想知道给定的 DAO 是否可以接受 Person 对象。这也可以通过 API 实现:

(new TypeToken<DAO<?, Person>>() {})
.isSupertypeOf(TypeToken.of(PersonDAOExtension.class))

这个和其他 Guava 反射实用程序的功能有一个体面的解释 on the Guava GitHub wiki .

关于java - 如何使用反射解析泛型返回类型的实际类型?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17297308/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com