Java反射的理解(六)-- 通过反射了解集合泛型的本质
Java反射的理解(六)-- 通过反射了解集合泛型的本质
上述写了那么多,我们可能会有个疑问,为什么要用反射,步骤比我们常规的加载类操作复杂多了,别急,这个问题我最后才解答,我们先来了解集合泛型的本质。
直接上代码:
import java.lang.reflect.Method;
import java.util.ArrayList;
public class MethodDemo4 {
public static void main(String[] args) {
ArrayList list = new ArrayList();
ArrayList<String> list1 = new ArrayList<String>();
list1.add("hello");
//list1.add(20);错误的
Class c1 = list.getClass();
Class c2 = list1.getClass();
System.out.println(c1 == c2);
//反射的操作都是编译之后的操作
/*
* c1==c2结果返回true说明编译之后集合的泛型是去泛型化的
* Java中集合的泛型,是防止错误输入的,只在编译阶段有效,
* 绕过编译就无效了
* 验证:我们可以通过方法的反射来操作,绕过编译
*/
try {
Method m = c2.getMethod("add", Object.class);
m.invoke(list1, 20);//绕过编译操作就绕过了泛型
System.out.println(list1.size());
System.out.println(list1);
/*for (String string : list1) {
System.out.println(string);
}*///现在不能这样遍历,遍历的时候 String,list 类型不一致,会报错,可以用 Object 父类型来遍历
} catch (Exception e) {
e.printStackTrace();
}
}
}
在刚开始中,如果我们尝试 list.add(20) 肯定会报错的,因为指定的 list 类型是 String,添加的 20 是 int 型,类型不一样。
但可以看到得到的类类型 c1 和 c2 是一样的,这说明编译之后集合的泛型是去泛型化的,所以我们可以通过方法的反射来操作,绕过编译,让 list.add(20) 成功,结果也证明这样是可行的。所以得出结论:
Java中集合的泛型,是防止错误输入的,只在编译阶段有效,绕过编译就无效了。
结论:
所以反射到底有什么用呢?
答:反射可以绕过编译阶段。可以动态加入代码,一个对象的方法未完全实现,也可以调用该方法(部分)。比如 Spring 中的 IOC 这块,就用到了反射知识。
完。