首页 > 作文

图文详解java反射机制及常用应用场景

更新时间:2023-04-06 02:09:05 阅读: 评论:0

一、什么是java反射?

在java的面向对象编程过程中,通常我们需要先知道一个class类,然后new 类名()方式来获取该类的对象。也就是说我们需要在写代码的时候(编译期或者类加载之前)就知道我们要实例化哪一个类,运行哪一个方法,这种通常被称为静态的类加载。

但是在有些场景下,我们事先是不知道我们的代码的具体行为的。比如,我们定义一个服务任务工作流,每一个服务任务都是对应的一个类的一个方法。

服务任务b执行哪一个类的哪一个方法,是由服务任务a的执行结果决定的服务任务c执行哪一个类的哪一个方法,是由服务任务a和b的执行结果决定的并且用户不希望服务任务的功能在代码中写死,希望通过配置的方式根据不同的条件执行不同的程序,条件本身也是变化的

面对这个情况,我们就不能用代码new 类名()来实现了,因为你不知道用户具体要怎么做配置,这一秒他希望服务任务a执行xxxx类的x方法,下一秒他可能希望执行yyyy类的y方法。当然你也可以说提需求嘛,用户改一次需求,我改一次代码。这种方式也能需求,但对于用户和程序员个人而言都是痛苦,那么有没有一种方法在运行期动态的改变程序的调用行为的方法呢?这就是要为大家介绍的“java反射机制”。

那么java的反射机制能够做那些事呢?大概是这样几种:

在程序运行期动态的根据package名.类名实例化类对象在程序运行期动态获取类对象的信息,包括对象的成本变量和方法在程序运行期动态使用对象的成员变量属性在程序运行期动态调用对象的方法(私有方法也可以调用)

二、hello world

先入个门,大佬可以略过这一段。我们定义一个类叫做student

如果不用反射的方式,我相信只要学过java的朋友肯定会调用dinner方法

如果是反射的方式我们该怎么调用呢?

通过上面的代码我们看到,com.zimug.java.reflection.student类名和dinner方法名是字符串。既然是字符串我们就可以通过配置文件,或数据库、或什么其他的灵活配置方法来执行这段程序了。这就是反射最基础的使用方式。

三、类加载与反射关系

java的类加载机制还是挺复杂的,我们这里为了不混淆重点,只为大家介绍和“反射”有关系的一部分内容。

java执行编译的时候将java文件编译成字节码class文件,类加载器在类加载阶段将class文件加载到内存,并实例化一个java.lang.class的对象。比如对于student类在加载阶段会有如下动作:

在内存(方法区或叫代码区)中实例化一个class对象,注意是class对象不是student对象一个class类(字节码文件)对应一个class对象,并且只有一个该class对象保存了student类的基础信息,比如这个student类有几个字段(filed)?有几个构造方法(constructor)?有几个方法(method)?有哪些注解(annotation)?等信息

有了上面的关于student类的基本信息对象(java.lang.class对象),在运行期就可以根据这些信息来实例化student类的对象。

在运行期你可以直接new一个student对象也可以使用反射的方法构造一个student对象

但是无论你new多少个student对象,不论你反射构建多少个student对象,保存student类信息的java.lang.class对象都只有一个。下面的代码可以证明。

四、操作反射的java类

了解了上面的这些基础信息,我们就可以更深入学习反射类相关的类和方法了:

java.lang.class: 代表一个类

java.lang.reflect.constructor: 代表类的构造方法

java.lang.reflect.method: 代表类的普通方法

java.lang.reflect.field: 代表类的成员变量

java.lang.reflect.modifier: 修饰符,方法的修饰符,成员变量的修饰符。

java.lang.annotation.annotation:在类、成员变量、构造方法、普通方法上都可以加注解

4.1.获取class对象的三种方法

class.forname()方法获取class对象

类名.class获取class对象

类对象.getclass()方式获取class对象

虽然有三种方法可以获取某个类的class对象,但是只有第一种可以被称为“反射”。

4.2.获取class类对象的基本信息

class类对象信息中几乎包括了所有的你想知道的关于这个类型定义的信息,更多的方法就不一一列举了。还可三七粉的功效与作用及正确吃法以通过下面的方法

获取class类对象代表的类实现了哪些接口: getinterfaces()

获取class类对象代表的类使用了哪些注解: getannotations()

4四年级数学上册教案.3. 获得class对象的成员变量

结合上文中的student类的定义理解下面的代码

getfields()方法获取类的非私有的成员变量,数组,包含从父类继承的成员变量

getdeclaredfields方法获取所有的成员变量,数组,但是不包含从父类继承而来的成员变量

4.4.获取class对象的方法

getmethods() : 获取class对象代表的类的所有的非私有方法,数组,包含从父类继承而来的方法

getd陕西师范大学在哪eclaredmethods() : 获取class对象代表的类定义的所有的方法,数组,但是不包含从父类继承而来的方法

getmethod(methodname): 获取class对象代表的类的指定方法名的非私有方法

getdeclaredmethod(methodname): 获取class对象代表的类的指定方法名的方法

上面代码的执行结果如下:

student对象的非私有方法
dinner,wait,wait,wait,equals,tostring,hashcode,getclass,notify,notifyall, end
student对象的所有方法
dinner,sleep, end
dinner方法的参数个数0
sleep方法的参数个数1
sleep方法的参数对象数组[int arg0]
sleep方法的参数返回值类型void

可以看到getmethods获取的方法中包含object父类中定义的方法,但是不包含本类中定义的私有方法sleep。另外我们还可以获取方法的参数及返回值信息:

获取参数相关的属性:

获取方法参数个数:getparametercount()获取方法参数数组对象:getparameters() ,返回值是java.lang.reflect.parameter数组

获取返回值相关的属性

获取方法返回值的数据类型:getreturntype()

4.5.方法的调用

实际在上文中已经演示了方法的调用,如下invoke调用dinner方法

dinner方法是无参的,那么有参数的方法怎么调用?看看invoke方法定义,第一个参数是method对象,无论后面object... args有多少参数就按照方法定义依次传参就可以了。

4.6.创建类的对象(实例化对象)

五、反射的常用场景

通过配置信息调用类的方法

结合注解实现特殊功能

按需加载jar包或class

5.1. 通过配置信息调用类的方法

将上文的hello world中的代码封装一下,你知道类名classname和方法名methodname是不是就可以调用方法了?至于你将classname和 methodname配置到文件,还是nacos,还是数据库,自己决定吧!

5.2.结合注解实现特殊功能

大家如果学习过mybatis plus都应该学习过这样的一个注解tablename,这个注解表示当前的实体类student对应的数据库中的哪一张表。如下问代码所示,student所示该类对应的是t_student这张表。

下面我们自定义tablename这个注解

有了这个注解,我们就可以扫描某个路径下的java文件,至于类注解的扫描我们就不用自己开发了,引入下面的maven坐标就可以

看下面代码:先扫描包,从包中获取标注了tablename注解的类,再对该类打印注解value信息

输出结果是:

com.zimug.java.reflection.student类,tablename注解value=t_student

有的朋友会问这有什么用?这有大用处了。有了类定义与数据库表的对应关系,你还能通过反射获取类的成员变量,之后你是不是就可以根据表明t_student和字段名nickname,age构建增删改查的sql了?全都构建完毕,是不是就是一个基础得mybatis plus了?

反射和56个民族56支花注解结合使用,可以演化出许许多多的应用场景,特别是在框架代码实现方面。等待你去发觉啊!

5.3.按需加载jar包或class

在某些场景下,我们可能不希望jvm的加载器一次性的把所有classpath下的jar包装载到jvm虚拟机中,因为这样会影响项目的启动和初始化效率,并且占用较多的内存。我们希望按需加载,需要用到哪些jar,按照程序动态运行的需求取加载这些jar。

把jar包放在classpath外面,指定加载路径,实现动态加载。

同样的把.class文件放在一个路径下,我们也是可以动态加载到的

类的动态加载能不能让你想到些什么?是不是可以实现代码修改,不需要重新启动web容器?对的,就是这个原理,因为一个类的class对象只有一个,所以不管你重新加载多少次,都是使用最后一次加载的class对象(上文讲过哦)。

六、反射的优缺点

优点:自由,使用灵活,不受类的访问权限限制。可以根据指定类名、方法名来实现方法调用,非常适合实现业务的灵活配置。在框架开发方面也有非常广泛的应用,特别是结合注解的使用。

缺点:

也正因为反射不受类的访问权限限制,其安全性低,很大部分的java安全问题都是反射导致的。相对于正常立春的时间的对象的访问调用,反射因为存在类和方法的实例化过程,性能也相对较低破坏java类封装性,类的信息隐藏性和边界被破坏

言尽于此,限于笔者的知识结构,可能有不严谨之处,欢迎大家讨论与指正!期待您的关注,我将持续带来更哇塞的作品,希望大家以后多多支持www.887551.com!

本文发布于:2023-04-06 02:09:04,感谢您对本站的认可!

本文链接:https://www.wtabcd.cn/fanwen/zuowen/32ff373c2f927cbf3553603bcd2fc665.html

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,我们将在24小时内删除。

本文word下载地址:图文详解java反射机制及常用应用场景.doc

本文 PDF 下载地址:图文详解java反射机制及常用应用场景.pdf

标签:方法   对象   反射   注解
相关文章
留言与评论(共有 0 条评论)
   
验证码:
Copyright ©2019-2022 Comsenz Inc.Powered by © 专利检索| 网站地图