Java反射机制
创始人
2024-05-09 05:21:10
0

目录

反射问题的引出

Java程序在计算机中部署的三个阶段

反射的主要相关类

反射机制的优缺点

调优

反射常用类—Class

特点

常用方法

获取映射Class类对象的四种方式

类加载的三个阶段

加载阶段 Loading

链接阶段 Linking

验证 Verification

准备 Preparation

 解析 Resolution

初始化阶段 Initialization

反射爆破

创建实例

操作属性

操作方法


反射问题的引出

读取一个下述配置文件中指定的信息,创建对象并且调用方法。

利用除反射之外的技术显然是无法在运行阶段动态加载Dog类并创建Dog实例对象的,但是利用Java的反射机制,就可以实现动态加载的效果,并且可以通过修改配置文件在不带便源码的情况下控制程序(例如将配置文件中的method换成别的方法名称,可以不改变源码而调用对象实例的更换的方法):
 

public class Reflect_ {public static void main(String[] args) {Properties pro = new Properties();try {pro.load(new FileInputStream("src\\test.properties"));String classPath = pro.getProperty("classPath");String methodName = pro.getProperty("method");//在运行时加载配置文件中Dog类的Class映射类Class cls = Class.forName(classPath);Object dog = cls.newInstance();   //获得一个Dog的实例对象//获得Dog类的映射Class对象中的方法对象Method method = cls.getMethod(methodName);//调用通过映射创建的Dog实例对象中的hi方法method.invoke(dog);} catch (IOException e) {e.printStackTrace();} catch (ClassNotFoundException e) {e.printStackTrace();} catch (InstantiationException e) {e.printStackTrace();} catch (IllegalAccessException e) {e.printStackTrace();} catch (NoSuchMethodException e) {e.printStackTrace();} catch (InvocationTargetException e) {e.printStackTrace();}}
}

Java程序在计算机中部署的三个阶段

在弄清楚反射机制前,需要先对Java程序的三个部署阶段(编译阶段、加载阶段、运行阶段)进行学习:


反射的主要相关类

与反向相关的类主要有四个:

  • java.lang.Class
    该对象表示某个类加载后在堆中的映射对象,包含了这个类的字段、方法信息
  • java.lang.reflect.Method
    该类包含了某个类在加载后的方法信息
  • java.lang.reflect.Field
    该类包含了某个类在加载后的字段信息
  • java.lang.reflect.Constructor
    该类包含了某个类在加载后的构造方法信息

反射机制的优缺点

优点:可以动态的加载使用对象(也是框架底层的核心),使用灵活
缺点:使用反射基本是解释执行,对程序的运行效率有影响

通过下述验证会发现用于反射动态加载类调用方法的效率大不如静态加载类:

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;public class Reflect_02 {public static void main(String[] args) throws ClassNotFoundException, InvocationTargetException, InstantiationException, IllegalAccessException, NoSuchMethodException {tradition();reflection();}//传统方法调用hipublic static void tradition() {Cat cat = new Cat();long start = System.currentTimeMillis();    //获取当前系统的相对时间System.out.println(start);for (int i = 0; i < 900000000; i++) {cat.hi();}long end = System.currentTimeMillis();System.out.println(end-start);}public static void reflection() throws ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchMethodException, InvocationTargetException {Class cls = Class.forName("com.shuai.Cat");Cat cat = (Cat)cls.newInstance();Method hi = cls.getMethod("hi");//关闭访问检测,优化反射效率hi.setAccessible(true); //取消反射调用方法时检查long start = System.currentTimeMillis();for (int i = 0; i < 900000000; i++) {hi.invoke(cat);}long end = System.currentTimeMillis();System.out.println(end-start);}
}//静态加载类的tradition方法的运行时间:4ms
//通过反射动态加载类的reflection方法的运行时间:576ms

调优

针对上述反射机制进行调优可以通过设置不进行反射检查,提高反射程序的运行效率:
在通过某个类的Class映射对象cls获得该类的方法或字段对象后,可以设置方法/字段.setAccessble(true);禁用访问安全检查的开关,提高程序的运行效率。


反射常用类—Class

特点

  • Class类也是一个类,继承自Object(易于其它类混淆)
  • Class类对象是在类加载阶段由系统创建的,而不是new出来的;并且Class类对象在内存(堆)中只存在一份,因为类只加载一次
  • 每个类的实例对象都会存储关于自己属于哪个Class映射类的信息,因此也都知道自己属于哪个Class类对象

常用方法

Class cls = Class.forName(classPath);
cls.toString();               //显示自己是哪个类的Class对象
cls.getClass();               //运行类型
cls.getPackage().getName();   //获得关联类的包信息
cls.getName();                //获得映射类的名称
cls.newInstance();            //通过反射创建映射类的实例对象
cls.getField(字段名);          //通过反射获取映射类的字段信息(private字段不能访问到)
field.set(通过反射创建的实例对象,值);    //通过反射给对象字段赋值
field.get(通过反射创建的实例对象);       //获取映射类的属性的值
cls.getMethod(方法名);         //通过反射获得映射类的成员方法信息
method.invoke(通过反射创建的对象);    //通过反射访问映射类的方法
clas.getConstructor(类.class,类.class...);    //获得映射类的构造方法信息

获取映射Class类对象的四种方式

  1. Class.forName(classPath);
    已知一个类的全名称并且该类在当前类路径下,可以使用Class类的静态方法Class.forName(classPath)加载classPath路径下的类的Class对象
    应用场景:多用于配置文件,读取类全路径,加载类
  2. 类名.class
    已知具体的类,可以通过 类.class获取,该方式最安全最可靠,程序的性能最高
    应用场景:多用于参数传递,比如通过反射得到相应的构造器对象,也可用于基本数据类型
  3. 类的实例.getClass( )
    已知某个类的实例,可以调用该实例对象的getClass方法获得该对象所属类的Class对象
    应用场景:实例对象获取类的Class对象
  4. 包装类.TYPE
    基本数据类型对应的包装类,可以通过类名.TYPE获得Class对象

类加载的三个阶段

加载阶段 Loading

JVM将类的字节码从不同的数据源(class文件,jar包,网络等)将数据转化为二进制字节流读入内存,并创建一个java.lang.Class对象

链接阶段 Linking

验证 Verification

确保Class文件的字节流中包含的信息符合当前虚拟机的要求,包括:文件格式验证(是否以魔数0x cafebabe开头)、元数据验证、字节码验证、符号引用验证等。如果项目较大,需要加载的类很多,在此阶段也可以通过 -Xverify:none参数关闭大部分的类的验证机制,缩短虚拟机类加载的时间达到优化效果。

准备 Preparation

JVM在该阶段会对静态变量进行内存分配和默认初始化(初始化为对应数据类型的默认值),静态变量所使用的内存将在方法区中进行分配。

 解析 Resolution

JVM将常量池内的符号引用替换为直接引用。

初始化阶段 Initialization

编译器按照代码语句在源文件中出现的顺序,依次自动收集类中所有的静态变量的赋值动作和静态代码块中的内容并进行合并生成方法,并且在此阶段JVM会保证一个类的方法在多线程的环境中会被正确加锁,同步,也就是说,如果有多个线程同时初始化一个类,那么只会有一个线程去执行这个类的方法,其他线程都会阻塞等待,直到活动线程执行方法结束


反射爆破

创建实例

通过反射创建实例有两种方式:

  • 直接调用获取到的反射对象的newInstance方法,该方法通过调用类的public的无参构造创建实例对象。
  • 通过getConstructor() 并且指定参数类型获取该类的公共构造器,通过指定构造器创建对象
  • 通过getDeclaredConstructor() 并且指定参数类型获取该类所有类型的构造器并设置反射爆破创建实例对象:
  • 例如:

操作属性

  • 通过getField()并且指定属性名称可以获取到该类中公共的属性
  • 通过getDeclaredField() 指定属性名称获取该类中所有类型的属性;并且设置反射爆破后,可以访问私有的属性
  • 例如:
     

操作方法

  • 通过getMethod()并且指定形参类型,可以获得类中公共的方法
  • 通过getDeclaredMethod()并指定形参类型可以获得类中所有类型的方法,在设置反射爆破后,可用于调用类中的私有方法。
  • 例如:

 

相关内容

热门资讯

晶采观察丨八部门发文!银发福利...   赶紧转给爸妈,2026年养老消费有了更多新选择。近日,民政部等八部门联合印发《关于培育养老服务经...
视频丨为夺格陵兰岛 美再挥关税...   17日,丹麦本土和格陵兰岛多地分别举行游行示威,抗议美国近期持续发表觊觎格陵兰岛的言论。然而就在...
哥伦比亚发生武装冲突 已致20...   当地时间16日,哥伦比亚中南部的瓜维亚雷省埃尔雷东托市农村地区发生一起严重武装冲突,造成至少20...
AI生成图片竟成人像摄影指南 ...   AI写诗、作画已经不新鲜了,但如果您买了一本教人摄影的专业书籍,回家却发现书里的模特长着六根手指...
欧盟-南共市自贸协定正式签署 ...   当地时间1月17日,欧盟-南方共同市场(南共市)自由贸易协定在巴拉圭首都亚松森正式签署。阿根廷总...
两部门安排部署重点省份做好低温...   针对1月17日至21日明显雨雪过程,1月17日,国家防灾减灾救灾委员会办公室、应急管理部组织中国...
澎湃“中国心”!看国产燃气轮机...   近5年突破200余项关键核心技术,创造工业燃气轮机领域多个“零”的突破;从单一型号到形成3至50...
赛事火热、“南客北上”、长期参...   央视网消息:近日,随着第十二届全国大众冰雪季的全面启动以及寒假的陆续到来,各主要冰雪运动目的地客...
遥远星系里“小红点”神秘天体是...   中新网北京1月17日电 (记者 孙自法)国际知名学术期刊《自然》最新发表一篇天文学论文称,研究人...
美舰船过航台湾海峡 东部战区全...   东部战区新闻发言人就美“菲恩”号导弹驱逐舰、“西尔斯”号海洋测量船过航台湾海峡发表谈话  东部战...