对于泛型类型的属性和方法可以通过Filed和Method在运行时获取泛型信息,但是如何在运行时获取泛型类的泛型信息呢?
我们知道java的泛型是伪泛型,只是在编译的时候进行类型检查,编译后,所有的泛型信息都会被擦除,因此在运行时无法使用反射获取泛型的类型。比如对于泛型类List<String>
,在编译的时候会检测所有加入到list里面的对象是否是String类型,编译后的class文件中是不会有泛型信息的。
那么该如何获取泛型类的泛型信息呢,可以通过子类继承泛型类的方法来实现。
public class GenericTest {
public static abstract class ParentClass<T> {
T name;
public T getName() {
return name;
}
public void setName(T name) {
this.name = name;
}
}
public static class Child extends ParentClass<String> {
}
/**
* 通过父类去获取泛型类型
*/
public static void getActualType() {
Child child = new Child();
Type type = child.getClass().getGenericSuperclass();
if(type instanceof ParameterizedType) {
ParameterizedType parameterizedType = (ParameterizedType) type;
Type actualtype = parameterizedType.getActualTypeArguments()[0];
System.out.println(actualtype);//class java.lang.String
}
}
public static void main(String[] args) {
getActualType();
}
}
java在编译的时候会检测父类的泛型信息,因为子类声明了泛型的具体类型,所以子类的class文件中记录该子类声明的泛型类型,所以只有在这种情况下才能在运行时通过反射API获取到该泛型的具体类型。
子类是如何存储泛型信息的?
上述例子中,子类Child的字节码文件中存储有String这部分泛型信息,而且这部分泛型信息是存储在被旧版本的JVM忽略的字节码区域中,所以旧版本的JVM仍然可以正常运行这种字节码。
ParentClass<String>
这种泛型类在编译时是会擦除所有泛型信息的,字节码中没有地方存储泛型信息,所以必须通过子类继承泛型类的方式在子类的字节码中存储泛型信息。
原始类型(Raw Types)
原始类型就是泛型类型在擦除类型信息后在字节码中的真正类型。
比如ArrayList<String>
、ArrayList<Integer>
擦除类型信息后都是ArrayList类型。
参考:
https://docs.oracle.com/javase/tutorial/java/generics/rawTypes.html