泛型是通过类型擦除来实现的,编译器在编译时擦除了所有类型相关的信息,所以在运行时不存在任何类型相关的信息。
当泛型被擦除后,他有两种转换方式,第一种是如果泛型没有设置类型上限,那么将泛型转化成Object类型,第二种是如果设置了类型上限,那么将泛型转化成他的类型上限。
例如 List 在运行时仅用一个List来表示。这样做的目的,是确保能和Java 5之前的版本开发二进制类库进行兼容。你无法在运行时访问到类型参数,因为编译器已经把泛型类型转换成了原始类型。
//未指定上限
public class Test1 {
T t;
public T getValue() {
return t;
}
public void setVale(T t) {
this.t = t;
}
}
//指定上限
public class Test2 {
T t;
public T getT() {
return t;
}
public void setT(T t) {
this.t = t;
}
}
//通过反射调用获取他们的属性类型
@Test
public void testType1() {
Test1 test1 = new Test1<>();
test1.setVale("11111");
Class extends Test1> aClass = test1.getClass();
for (Field field : aClass.getDeclaredFields()) {
System.out.println("Test1属性:" + field.getName() + "的类型为:" + field.getType().getName());
}
Test2 test2 = new Test2();
test2.setT(new ArrayList());
Class extends Test2> aClass2 = test2.getClass();
for (Field field : aClass2.getDeclaredFields()) {
System.out.println("test2属性:" + field.getName() + "的类型为:" + field.getType().getName());
}
}
上面方法打印的结果:
Test1属性:t的类型为:java.lang.Object
Test1属性:this$0的类型为:com.company.genericity
test2属性:t的类型为:java.util.List
test2属性:this$0的类型为:com.company.genericity