黑马程序员学习笔记——泛型的反射

来源:互联网 发布:数控车编程培训 编辑:程序博客网 时间:2024/05/20 23:36

---------------------- 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

0 0
原创粉丝点击