zoukankan      html  css  js  c++  java
  • 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 这块,就用到了反射知识。


    完。

  • 相关阅读:
    hdfs shell命令
    雪碧图
    绝对定位
    相对定位
    Vue 自定义指令
    Vue 【组件】组件注册、组件生命周期、动态组件、keep-alive
    Git 使用
    React 【生命周期】三个阶段生命周期函数、不同生命周期详解、图解生命周期
    【华为云技术分享】一统江湖大前端DOClever—你的Postman有点Low
    【华为云技术分享】圣诞特别版 | 数据库频频出现OOM问题该如何化解?
  • 原文地址:https://www.cnblogs.com/weixuqin/p/11220750.html
Copyright © 2011-2022 走看看