---------------------- ASP.Net+Unity开发、.Net培训、期待与您交流! ----------------------
如果获取一个泛型的反射呢?
分析如下:
首先,以前认为泛型是给编译器看的,编译好后不会在字节码文件中保留泛型的类型信息。
这个理解有一点偏差,纠正如下:
泛型信息在编译后是会保存到字节码文件中的,但并不是在类的字节码中,也就是说,字节码文件中不仅有类,也具有类的其他信息。
而java类装载器在装载字节码文件的时候单独装载了类(这样性能更高),而把类的泛型信息保存到了其他地方,所以在反射的时候还是能反射到泛型,但类上确实没有了泛型信息。
其次,比如ArrayList<String> al = new ArrayList<String>();
如果想要得到al的泛型信息,如果不做任何改变是不可能的。至于为什么在编译是JVM就能判定非String类型不能存入,这时因为这段方法的局部变量保存区(stack中)中保存了相关信息。
我们可以新定义一个类,让它从ArrayList<String>继承,然后根据这个新类来获取泛型信息,代码如下:
package ProxyGenericText;
import java.lang.reflect.Method;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.List;
public class Test1 {
public static void main(String[] args) throws Exception {
//第一种方法,用方法上的参数来获取泛型
Class c1 = Test1.class;
Method m = c1.getMethod("mm", List.class);
Type[] types = m.getGenericParameterTypes();
ParameterizedType pt = (ParameterizedType)types[0];
System.out.println("原类型"+pt.getRawType());
Type[] types1 = pt.getActualTypeArguments();
System.out.println("实际类型" +types1[0]);
System.out.println("第二种形式,类的参数化------");
//第二种方法,用类申明上的信息来获取泛型
MyList my = new MyList();
Class c2 = my.getClass();
Type t2 = c2.getGenericSuperclass();
ParameterizedType pt2 = (ParameterizedType)t2;
Type[] types2 = pt2.getActualTypeArguments();
System.out.println("原类型" +pt2.getRawType());
System.out.println("实际类型1"+t2);
System.out.println("实际类型2"+types2[0]);
}
public static void mm(List<Integer> lt) {
}
}
class MyList extends ArrayList<Long> {
}
补充:
可以做一个实验,写一个类A和B,如下:
public class A<T> {
}
public class B {
}
注意,A上有泛形,B上没有,编译后,看它们生成的字节码文件大小:
明显2个字节码文件大小不同(如果A上没有泛形,它们的文件大小是一样的)。这样可以得出一个结论,泛形信息编译后其实是会保存到字节码文件中的。
张老师没说错,他应该是说类上没有泛形了,也就是说字节码文件中不仅有类,也有类的其它信息。java的类装载器在装载字节码文件时单独装载了类(这是出于性能方面的考虑),而把类的泛形信息保存到了其它的地方,所以你在反射时还是反射得到的。但类上的确没有了泛形信息,所以我们会经常看到“擦除”这个词。
---------------------- ASP.Net+Unity开发、.Net培训、期待与您交流! ----------------------详细请查看:http://edu.csdn.net