Java的泛型是使用的擦除法实现,泛型的定义只在编译的时候有效,编译之后是没有保留泛型的类型信息的。
然而,擦除法的实现存在一些特列,在这些特例情况下,Java会记录泛型的类型信息,并且可以通过反射的Api来获取。
一)泛型继承
public class TypeTest {
public static void main(String[] args) {
doProxy(Lists.newArrayList(1), new Function() {
public List execute(Integer request) {
return Lists.newArrayList(String.valueOf(request));
}
});
}
//必须是class,interface获取不到泛型
abstract static class Function {
abstract List execute(F request);
}
public static List doProxy(List request, Function function) {
if (function.getClass().getGenericSuperclass() instanceof ParameterizedType) {
Type mySuperClass = function.getClass().getGenericSuperclass();
Type type = ((ParameterizedType) mySuperClass).getActualTypeArguments()[0];
System.out.println(type);
System.out.println(((ParameterizedType) mySuperClass).getActualTypeArguments()[1]);
}
List list = Lists.newArrayList();
for (F f : request) {
list.addAll(function.execute(f));
}
return list;
}
}
二)方法参数和返回值的泛型
因为需要先通过反射获取Method对象,而反射获取Method对象需要方法名和参数类型,所以
只能获取List method(Set param),这种参数的泛型
不能获取T method(T param),这种参数的泛型
method.getGenericParameterTypes()[0].getActualTypeArguments()[0]
method.getGenericReturnType().getActualTypeArguments()[0]
三)Field的泛型
同方法泛型的原因
只能获取List field,这种属性的泛型
不能获取T field这种属性的泛型
field.getGenericType().getActualTypeArguments()[0]