反射
#2020云栖大会#阿里云海量offer来啦!投简历、赢阿里云限量礼品及阿里云ACA认证免费考试资格!>>>
反射
IT那个小笔记
类的加载时机
类加载器
什么是反射
通过字节码来使用
越过数组泛型检测
类的加载时机
当程序要使用某个类时,如果该类还未被加载到内存中,系统会通过加载,连接,初始化三步来实现对这个类进行初始化
加载
就是指将class文件读入内存,并为之创建一个Class对象。任何类被使用时系统都会建立一个Class对象。
连接
验证 是否有正确的内部结构,并和其他类协调一致
准备 负责为类的静态成员分配内存,并设置默认初始化值
初始化
初始化成员变量等等
类什么时候会被加载
创建类的实例
调用类的静态方法、访问静态变量
初始化某个类的子类,就会也加载父类
使用反射方式来强制创建某个类或接口对应的java.lang.Class对象
类加载器
什么是类加载器classLoader
负责将.class文件加载到内存中,并为之生成对应的Class对象。虽然我们不需要关心类加载机制,但是了解这个机制我们就能更好的理解程序的运行。
类加载器分类
根类加载器
也被称为引导类加载器,负责Java核心类的加载
比如System,String等。在JDK中JRE的lib目录下rt.jar文件中
扩展类加载器
负责JRE的扩展目录中jar包的加载。
在JDK中JRE的lib目录下ext目录
系统类加载器
负责在JVM启动时加载来自java命令的class文件
以及classpath环境变量所指定的jar包和类路径
什么是反射
创建一个对象的三个阶段
源文件阶段 .java的文件
字节码阶段 .class
创建对象阶段 new 对象名称
内省
在运行时能够获取Bean类当中的属性名称和get与set方法,可以去遍历属性,故在编写DBUtils工具类会用到,获取任意不同domain类的属性信息
反射
JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;
对于任意一个对象,都能够调用它的任意一个方法和属性;
这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
无论是内省还是反射,首先都要获取字节码文件
获取字节码文件的三种方式
/* 第一种用Object类的getClass方法 */Test t = new Test();Class tc = t.getClass();/* 第二种用Class类的静态方法forName */Class tc = Class.forName("Test类的全限定名称");/* 第三种通过静态属性class */Class tc =Test.class;
字节码使用
用字节码来创建对象
1.通过无参构造创建对象
获取字节码
调用字节码的newInstance()方法
Class tc = Test.class;Test t = (Test)tc.newInstance()
2.通过有参构造创建对象
获取字节码的构造器
Constructor c = tc.getConstructor(Integer.class,String.class);Test t = (Test)c.newInstance(18,"张三");
因为在反射阶段操作的都是字节码,不知道具体的类型,只有在创建对象的时候才去给实际参数
用字节码获取字段
/* 获取公共字段 */Class tc = Test.class;Test t = (Test)tc.newInstance();Field f = tc.getField("字段名");// 获取字段f.set(t,值);// 给某个对象的此字段设值/* 获取私有字段 */Class tc = Test.class;Test t = (Test)tc.newInstance();Field f = tc.getDeclareField("字段名");// 获取私有字段f.setAccessible(true);// 将私有解开才能访问进去f.set(t,值);
获取方法也是一样
/* 获取方法 */Method m = tc.getMethod("方法名");m.invoke(对象);/* 有参 */Method m = tc.getMethod("方法名",Integer.class,String.class);m.invoke(对象,16,"张三");/* 私有 */Method m = tc.getDeclareMethod("方法名");m.setAccessible(true);m.invoke(对象);
越过数组的泛型检查
数组如果定义好了泛型就不能添加泛型以外的类型
可以通过反射来去实现添加以外的类型
在一个Integer泛型的数组当中添加字符串类型
/* */ArrayList<Integer> list = new ArrayList<>();list.add(1);/*设置泛型后就只能添加相应类型的元素否则编译器就会提示错误,不能去编译但我们知道泛型,实际上在字节码中并不存在仅仅是在编译中的语法,让你遵守不能添加别的在运行时实际上并没有规定,只是你这样写了不然通不过编译所以我们可以通过获取字节码来跨过泛型*/Class ac = Class.forName("java.util.ArrayList");// 获取列表集合的字节码Method m = ac.getMethod("add",Object.class);// 获取add方法还有一个参数类型填Objectm.invoke(list,"张三");// 这个时候就添加上去了/*虽然list对象在被创建时设了泛型为整数,但只是在编译前、代码编写时有效实际上并不存在,所以通过字节码获取后就可以添加了*/
本文分享自微信公众号 - IT那个小笔记(qq1839646816)。
如有侵权,请联系 support@oschina.cn 删除。
本文参与“OSC源创计划”,欢迎正在阅读的你也加入,一起分享。

浙公网安备 33010602011771号