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 这块,就用到了反射知识。


完。

posted @ 2019-07-21 11:56  希希里之海  阅读(744)  评论(0编辑  收藏  举报