集合泛型:
可以放任何对象的 ArrayList ArrayList list = new ArrayList(); 没有限定泛型类型.
list.add("lilin");list.add(100);System.out.println(list);//很容得知结果能够是["lilin",100];
ArrayList<String> list1 = new ArrayList<String>(); 限定了只能存放String类型的数据
list1.add("lilin");
// String 泛型的集合 是不能添加int的类型的数据的
// list1.add(100);//此时编译是不会通过的,泛型检查了输入的正确性:
泛型的检查,只会在编译的时候检查,如果我们能够绕过编译的动作,就能避开输入的正确的检查操作,导致String泛型的集合中会存在不是String类型的数据.
反射reflect的操作,无论是CLass Method Filed 等,都是运行时加载的,不需要通过编译操作.
//下面通过反射操作,实现把int 100 放入 String泛型的list中去:
ArrayList<String> list = new ArrayList<String>(); Class c = list.getClass(); try {
//通过反射,获取到list的add方法,简单处理异常信息 Method m = c2.getMethod("add", Object.class);
//通过方法的反射,就能把int类型的数据,添加到string泛型的list中去 m.invoke(list1, 100);
//结果很明显,100 能够成功的添加进去 System.out.println(list1);
//此时要特别注意的:不能在用String 类型来for循环当前的list; 在循环到int的100会跑出类型不匹配的异常信息 for (String s : list1) { System.out.println(s); } } catch (Exception e) { e.printStackTrace(); }
其实:本质的原因是,泛型主要是控制输入的正确验证的,而编译后的泛型集合类型,也就是 Class c1 = list.getClass(); Class c2 = list1.getClass();
在编译后,集合的泛型是去泛型的,通过验证System.out.println(c1 == c2); 结果是true,也就是集合编译后,都是同一个类类型.
所以:集合的泛型和反射的相关的方法结合使用时,要特别注意,反射的是运行时加载的,集合的编译检查输入的正确是会被绕过的.