JAVA反射机制详解,一学就会
bigegpt 2025-05-15 16:31 15 浏览
目录
何为反射?
反射(Reflection),是指Java程序具有 在运行期 分析类以及修改其本身状态或行为的能力 。
通俗点说 就是 通过反射我们可以 动态地 获取一个类的所有属性和方法,还可以操作这些方法和属性。
实例的创建
一般我们创建一个对象实例 Person zhang = new Person();
虽然是简简单单一句,但JVM内部的实现过程是复杂的:
- 将硬盘上指定位置的Person.class文件加载进内存
- 执行main方法时,在栈内存中开辟了main方法的空间(压栈-进栈),然后在main方法的栈区分配了一个变量zhang。
- 执行new,在堆内存中开辟一个 实体类的 空间,分配了一个内存首地址值
- 调用该实体类对应的构造函数,进行初始化(如果没有构造函数,Java会补上一个默认构造函数)。
- 将实体类的 首地址赋值给zhang,变量zhang就引用了该实体。(指向了该对象)
其中上图步骤1 Classloader(类加载器) 将class文件加载到内存中具体分为3个阶段:加载、连接、初始化
而又在 加载阶段,类加载器 会 将类对应的.class文件中的二进制字节流读入到内存中,将这个字节流转化为方法区的运行时数据结构,然后在堆区创建一个**java.lang.Class 对象**(类相关的信息),作为对方法区中这些数据的访问入口
然后再通过 类的实例 来执操作 类的方法和属性 ,比如 zhang.eat(), zhang.getHeight() 等等
如果我们使用反射的话,我们需要拿到该类Person的Class对象,再通过Class对象来操作 类的方法和属性或者创建类的实例
Class personClass = Person.class;//这边只是举一个例子,获取class对象的多种方式,本文后面再慢慢道来
Object person = personClass.newInstance();
我们可以发现 通过new创建类的实例和反射创建类的实例,都绕不开.class文件 和 Class类的。
.class文件
首先我们得先了解一下 什么是.class文件
举个简单的例子,创建一个Person类:
public class Person {
/**
* 状态 or 属性
*/
String name;//姓名
String sex;//性别
int height;//身高
int weight;//体重
/**
* 行为
*/
public void sleep(){
System.out.println(this.name+"--"+ "睡觉");
}
public void eat(){
System.out.println("吃饭");
}
public void Dance(){
System.out.println("跳舞");
}
}
我们执行javac命令,编译生成Person.class文件
然后我们通过vim 16进制 打开它
#打开file文件
vim Person.class
#在命令模式下输入.. 以16进制显示
:%!xxd
#在命令模式下输入.. 切换回默认显示
:%!xxd -r
不同的操作系统,不同的 CPU 具有不同的指令集,JAVA能做到平台无关性,依靠的就是 Java 虚拟机。
.java源码是给人类读的,而 .class字节码是给JVM虚拟机读的 ,计算机智能识别 0 和 1组成的二进制文件,所以虚拟机就是我们编写的代码和计算机之间的桥梁。
虚拟机将我们编写的 .java 源程序文件编译为 字节码 格式的 .class 文件,字节码是各种虚拟机与所有平台统一使用的程序存储格式,class文件主要用于解决平台无关性的中间文件
Person.class文件 包含Person类的所有信息
Class类
我们来看下jdk的官方api文档对其的定义:
Class类的类表示正在运行的Java应用程序中的类和接口。 枚举是一种类,一个注释是一种界面。 每个数组也属于一个反映为类对象的类,该对象由具有相同元素类型和维数的所有数组共享。
原始Java类型( boolean , byte , char , short , int , long , float和double ),和关键字void也表示为类对象。
类没有公共构造函数。 相反, 类对象由Java虚拟机自动构建,因为加载了类,并且通过调用类加载器中的defineClass方法。。
**java 万物皆是Class类 **
【图片】
我们来看下Class类的源码,源码太多了,挑了几个重点:
public final class Class<T> implements java.io.Serializable,
GenericDeclaration,
Type,
AnnotatedElement {
private static final int ANNOTATION= 0x00002000;
private static final int ENUM = 0x00004000;
private static final int SYNTHETIC = 0x00001000;
private static native void registerNatives();
static {
registerNatives();
}
/*
* Private constructor. Only the Java Virtual Machine creates Class objects.
* This constructor is not used and prevents the default constructor being
* generated.
*/
private Class(ClassLoader loader) { //私有化的 构造器
// Initialize final field for classLoader. The initialization value of non-null
// prevents future JIT optimizations from assuming this final field is null.
classLoader = loader;
}
...
// reflection data that might get invalidated when JVM TI RedefineClasses() is called
private static class ReflectionData<T> {
volatile Field[] declaredFields;//字段
volatile Field[] publicFields;
volatile Method[] declaredMethods;//方法
volatile Method[] publicMethods;
volatile Constructor<T>[] declaredConstructors;//构造器
volatile Constructor<T>[] publicConstructors;
// Intermediate results for getFields and getMethods
volatile Field[] declaredPublicFields;
volatile Method[] declaredPublicMethods;
volatile Class<?>[] interfaces;//接口
// Value of classRedefinedCount when we created this ReflectionData instance
final int redefinedCount;
ReflectionData(int redefinedCount) {
this.redefinedCount = redefinedCount;
}
}
...
//注释数据
private volatile transient AnnotationData annotationData;
private AnnotationData annotationData() {
while (true) { // retry loop
AnnotationData annotationData = this.annotationData;
int classRedefinedCount = this.classRedefinedCount;
if (annotationData != null &&
annotationData.redefinedCount == classRedefinedCount) {
return annotationData;
}
// null or stale annotationData -> optimistically create new instance
AnnotationData newAnnotationData = createAnnotationData(classRedefinedCount);
// try to install it
if (Atomic.casAnnotationData(this, annotationData, newAnnotationData)) {
// successfully installed new AnnotationData
return newAnnotationData;
}
}
}
...
我们可以发现Class也是类,是一种特殊的类,将我们定义普通类的共同的部分进行抽象,保存类的属性,方法,构造方法,类名、包名、父类,注解等和类相关的信息。
Class类的构造方法是private, 只有JVM能创建Class实例 ,我们开发人员 是无法创建Class实例的,JVM在构造Class对象时,需要传入一个 类加载器 。
类也是可以用来存储数据的,Class类就像 普通类的模板 一样,用来保存“类所有相关信息”的类。
我们来继续看这个利用反射的例子: Class personClass = Person.class;
由于JVM为加载的 Person.class创建了对应的Class实例,并在该实例中保存了该 Person.class的所有信息,因此,如果获取了Class实例(personClass ),我们就可以通过这个Class实例获取到该实例对应的 Person类 的所有信息。
反射的使用
获取Class实例4种方式
- 通过对象调用 getClass() 方法来获取
Person p1 = new Person();
Class c1 = p1.getClass();
像这种已经创建了对象的,再去进行反射的话,有点多此一举。
一般是用于传过来的是Object类型的对象,不知道具体是什么类,再用这种方式比较靠谱
- 类名.class
Class c2 = Person.class;
这种需要提前知道导入类的包,程序性能更高,比较常用,通过此方式获取 Class 对象 ,Person类不会进行初始化
- 通过 Class 对象的 forName() 静态方法来获取,最常用的一种方式
Class c3 = Class.forName("com.zj.demotest.domain.Person");
这种只需传入类的全路径 , Class.forName会进行初始化initialization步骤 ,即静态初始化(会初始化类变量,静态代码块)。
- 通过类加载器对象的 loadClass() 方法
public class TestReflection {
public static void main(String[] args) throws ClassNotFoundException {
Person p1 = new Person();
Class c1 = p1.getClass();
Class c2 = Person.class;
Class c3 = Class.forName("com.zj.demotest.domain.Person");
//第4中方式,类加载器
ClassLoader classLoader = TestReflection.class.getClassLoader();
Class c4 = classLoader.loadClass("com.zj.demotest.domain.Person");
System.out.println(c1.equals(c2));
System.out.println(c2.equals(c3));
System.out.println(c3.equals(c4));
System.out.println(c1.equals(c4));
}
}
loadClass的源码:
public Class<?> loadClass(String name) throws ClassNotFoundException {
return loadClass(name, false);
}
loadClass 传入的第二个参数是"false",因此它不会对类进行连接这一步骤,根据 类的生命周期 我们知道,如果一个类没有进行验证和准备的话,是无法进行初始化过程的,即 不会进行类初始化,静态代码块和静态对象也不会得到执行
我们将c1,c2,c3,c4进行 equals 比较
System.out.println(c1.equals(c2));
System.out.println(c2.equals(c3));
System.out.println(c3.equals(c4));
System.out.println(c1.equals(c4));
结果:
true true true true
因为Class实例在JVM中是唯一的,所以,上述方法获取的Class实例是同一个实例, 一个类在 JVM 中只会有一个 Class 实例
Class类常用的API
日常开发的时候,我们一般使用反射是为了 创建类实例(对象)、反射获取类的属性和调用类的方法
getName() | 获得类的完整名字 |
getFields() | 获得类的public类型的属性 |
getDeclaredFields() | 获得类的所有属性。包括 private 声明的和继承类 |
getMethods() | 获得类的public类型的方法 |
getDeclaredMethods() | 获得类的所有方法。包括 private 声明的和继承类 |
getMethod(String name, Class[] parameterTypes) | 获得类的特定方法,name参数指定方法的名字,parameterTypes 参数指定方法的参数类型。 |
getConstructors() | 获得类的public类型的构造方法 |
getConstructor(Class[] parameterTypes) | 获得类的特定构造方法,parameterTypes 参数指定构造方法的参数类型 |
newInstance() | 通过类的不带参数的构造方法创建这个类的一个对象 |
getSuperClass() | 用于返回表示该 Class 表示的任何类、接口、原始类型或任何 void 类型的 超类的Class(即父类) 。 |
... | ... |
我们这边就不全部展开讲了,挑几个重点讲解一下
创建对象
- 调用class对象的 newInstance() 方法
Class c1 = Class.forName("com.zj.demotest.domain.Person");
Person p1 = (Person) c1.newInstance();
p1.eat();
结果:
吃饭
注意:Person类必须有一个 无参的构造器 且 类的构造器的访问权限不能是private
- 使用指定构造方法 Constructor 来创建对象
如果我们非得让Person类的无参构造器设为private呢,我们可以获取对应的Constructor来创建对象
Class c1 = Class.forName("com.zj.demotest.domain.Person");
Constructor<Person> con = c1.getDeclaredConstructor();
con.setAccessible(true);//允许访问
Person p1 = con.newInstance();
p1.eat();
结果:
吃饭
注意:setAccessible()方法能在运行时 压制 Java语言访问控制检查(Java language access control checks),从而能任意调用 被私有化 保护的方法、域和构造方法。
由此我们可以发现** 单例模式不再安全,反射可破之!**
访问属性
Field getField(name) | 根据字段名获取某个public的field(包括父类) |
Field getDeclaredField(name) | 根据字段名获取当前类的某个field(不包括父类) |
Field[] getFields() | 获取所有public的field(包括父类) |
Field[] getDeclaredFields() | 获取当前类的所有field(不包括父类) |
我们来看一个例子:
public class TestReflection3 {
public static void main(String[] args) throws Exception {
Object p = new Student("li hua");
Class c = p.getClass();
Field f = c.getDeclaredField("name");//获取属性
f.setAccessible(true);//允许访问
Object val= f.get(p);
System.out.println(val);
}
static class Student {
private String name;
public Student(String name) {
this.name = name;
}
}
}
结果:
li hua
我们可以发现 反射可以破坏类的封装
调用方法
Method getMethod(name, Class...) | 获取某个public的Method(包括父类) |
Method getDeclaredMethod(name, Class...) | 获取当前类的某个Method(不包括父类) |
Method[] getMethods() | 获取所有public的Method(包括父类) |
Method[] getDeclaredMethods() | 获取当前类的所有Method(不包括父类) |
我们来看一个例子:
public class TestReflection4 {
public static void main(String[] args) throws Exception {
//获取私有方法,需要传参:方法名和参数
Method h = Student.class.getDeclaredMethod("setName",String.class);
h.setAccessible(true);
Student s1 =new Student();
System.out.println(s1.name);
//传入目标对象,调用对应的方法
h.invoke(s1,"xiao ming");
System.out.println(s1.name);
}
static class Student {
private String name;
private void setName(String name) {
this.name = name;
}
}
}
结果:
null xiao ming
我们发现获取方法getMethod()时,需要传参 方法名和参数
这是因为.class文件中通常有 不止一个方法 ,获取方法getMethod()时,会去调用searchMethods方法循环遍历所有Method,然后根据 方法名和参数类型 找到唯一符合的Method返回。
我们知道类的方法是在JVM的方法区中 ,当我们new 多个对象时,属性会另外开辟堆空间存放,而方法只有一份,不会额外消耗内存,方法就像一套指令模板,谁都可以传入数据交给它执行,然后得到对应执行结果。
method.invoke(obj, args) 时传入目标对象,即可调用对应对象的方法
如果获取到的Method表示一个静态方法,调用静态方法时, 无需指定实例对象 ,所以invoke方法传入的第一个参数永远为null, method.invoke(null, args)
那如果 方法重写了呢, 反射依旧遵循 多态 的原则 。
反射的应用场景
如果平时我们只是写业务代码,很少会接触到直接使用反射机制的场景,毕竟我们可以直接new一个对象,性能比还反射要高。
但如果我们是工具框架的开发者,那一定非常熟悉,像 Spring/Spring Boot、MyBatis 等等框架中都大量使用反射机制, 反射被称为框架的灵魂
比如:
- Mybatis Plus可以让我们只写接口,不写实现类,就可以执行SQL
- 开发项目时,切换不同的数据库只需更改配置文件即可
- 类上加上@Component注解,Spring就帮我们创建对象
- 在Spring我们只需 @Value注解就读取到配置文件中的值
- 等等
扩展:反射配置文件
我们来模拟一个配置高于编码的例子
新建my.properties,将其放在resources的目录下
#Person类的包路径
className=com.zj.demotest.domain.Person
methodName=eat
Person类 还是本文 一直用的,在文章的开头有
最后我们来编写一个测试类
public class TestProp {
public static void main(String[] args) throws IOException, ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException {
Properties properties = new Properties();
ClassLoader classLoader = TestProp.class.getClassLoader();
InputStream inputStream = classLoader.getResourceAsStream("my.properties");// 加载配置文件
properties.load(inputStream);
String className = properties.getProperty("className");
System.out.println("配置文件中的内容:className="+className);
String methodName = properties.getProperty("methodName");
System.out.println("配置文件中的内容:methodName="+methodName);
Class name = Class.forName(className);
Object object = name.newInstance();
Method method = name.getMethod(methodName);
method.invoke(object);
}
}
结果:
配置文件中的内容:className=
com.zj.demotest.domain.Person
配置文件中的内容:methodName=eat
吃饭
紧接着,我们修改配置文件:
className=com.zj.demotest.domain.Person
methodName=eat
结果变为:
配置文件中的内容:className=
com.zj.demotest.domain.Person
配置文件中的内容:methodName=Dance
跳舞
是不是很方便?
尾语
反射机制是一种功能强大的机制,让Java程序具有在 运行期 分析类以及修改其本身状态或行为的能力 。
对于特定的复杂系统编程任务,它是非常必要的,为各种框架提供开箱即用的功能提供了便利,为解耦合提供了保障机制。
但是世事无绝对,反射相当于一系列解释操作,通知 JVM 要做的事情,性能比直接访问对象要差点(JIT优化后,对于框架来说实际是影响不大的),还会增加程序的复杂性等(明明直接new一下就能解决的事情,非要写一大段代码)。
相关推荐
- 当Frida来“敲”门(frida是什么)
-
0x1渗透测试瓶颈目前,碰到越来越多的大客户都会将核心资产业务集中在统一的APP上,或者对自己比较重要的APP,如自己的主业务,办公APP进行加壳,流量加密,投入了很多精力在移动端的防护上。而现在挖...
- 服务端性能测试实战3-性能测试脚本开发
-
前言在前面的两篇文章中,我们分别介绍了性能测试的理论知识以及性能测试计划制定,本篇文章将重点介绍性能测试脚本开发。脚本开发将分为两个阶段:阶段一:了解各个接口的入参、出参,使用Python代码模拟前端...
- Springboot整合Apache Ftpserver拓展功能及业务讲解(三)
-
今日分享每天分享技术实战干货,技术在于积累和收藏,希望可以帮助到您,同时也希望获得您的支持和关注。架构开源地址:https://gitee.com/msxyspringboot整合Ftpserver参...
- Linux和Windows下:Python Crypto模块安装方式区别
-
一、Linux环境下:fromCrypto.SignatureimportPKCS1_v1_5如果导包报错:ImportError:Nomodulenamed'Crypt...
- Python 3 加密简介(python des加密解密)
-
Python3的标准库中是没多少用来解决加密的,不过却有用于处理哈希的库。在这里我们会对其进行一个简单的介绍,但重点会放在两个第三方的软件包:PyCrypto和cryptography上,我...
- 怎样从零开始编译一个魔兽世界开源服务端Windows
-
第二章:编译和安装我是艾西,上期我们讲述到编译一个魔兽世界开源服务端环境准备,那么今天跟大家聊聊怎么编译和安装我们直接进入正题(上一章没有看到的小伙伴可以点我主页查看)编译服务端:在D盘新建一个文件夹...
- 附1-Conda部署安装及基本使用(conda安装教程)
-
Windows环境安装安装介质下载下载地址:https://www.anaconda.com/products/individual安装Anaconda安装时,选择自定义安装,选择自定义安装路径:配置...
- 如何配置全世界最小的 MySQL 服务器
-
配置全世界最小的MySQL服务器——如何在一块IntelEdison为控制板上安装一个MySQL服务器。介绍在我最近的一篇博文中,物联网,消息以及MySQL,我展示了如果Partic...
- 如何使用Github Action来自动化编译PolarDB-PG数据库
-
随着PolarDB在国产数据库领域荣膺桂冠并持续获得广泛认可,越来越多的学生和技术爱好者开始关注并涉足这款由阿里巴巴集团倾力打造且性能卓越的关系型云原生数据库。有很多同学想要上手尝试,却卡在了编译数据...
- 面向NDK开发者的Android 7.0变更(ndk android.mk)
-
订阅Google官方微信公众号:谷歌开发者。与谷歌一起创造未来!受Android平台其他改进的影响,为了方便加载本机代码,AndroidM和N中的动态链接器对编写整洁且跨平台兼容的本机...
- 信创改造--人大金仓(Kingbase)数据库安装、备份恢复的问题纪要
-
问题一:在安装KingbaseES时,安装用户对于安装路径需有“读”、“写”、“执行”的权限。在Linux系统中,需要以非root用户执行安装程序,且该用户要有标准的home目录,您可...
- OpenSSH 安全漏洞,修补操作一手掌握
-
1.漏洞概述近日,国家信息安全漏洞库(CNNVD)收到关于OpenSSH安全漏洞(CNNVD-202407-017、CVE-2024-6387)情况的报送。攻击者可以利用该漏洞在无需认证的情况下,通...
- Linux:lsof命令详解(linux lsof命令详解)
-
介绍欢迎来到这篇博客。在这篇博客中,我们将学习Unix/Linux系统上的lsof命令行工具。命令行工具是您使用CLI(命令行界面)而不是GUI(图形用户界面)运行的程序或工具。lsoflsof代表&...
- 幻隐说固态第一期:固态硬盘接口类别
-
前排声明所有信息来源于网络收集,如有错误请评论区指出更正。废话不多说,目前固态硬盘接口按速度由慢到快分有这几类:SATA、mSATA、SATAExpress、PCI-E、m.2、u.2。下面我们来...
- 新品轰炸 影驰SSD多款产品登Computex
-
分享泡泡网SSD固态硬盘频道6月6日台北电脑展作为全球第二、亚洲最大的3C/IT产业链专业展,吸引了众多IT厂商和全球各地媒体的热烈关注,全球存储新势力—影驰,也积极参与其中,为广大玩家朋友带来了...
- 一周热门
- 最近发表
-
- 当Frida来“敲”门(frida是什么)
- 服务端性能测试实战3-性能测试脚本开发
- Springboot整合Apache Ftpserver拓展功能及业务讲解(三)
- Linux和Windows下:Python Crypto模块安装方式区别
- Python 3 加密简介(python des加密解密)
- 怎样从零开始编译一个魔兽世界开源服务端Windows
- 附1-Conda部署安装及基本使用(conda安装教程)
- 如何配置全世界最小的 MySQL 服务器
- 如何使用Github Action来自动化编译PolarDB-PG数据库
- 面向NDK开发者的Android 7.0变更(ndk android.mk)
- 标签列表
-
- mybatiscollection (79)
- mqtt服务器 (88)
- keyerror (78)
- c#map (65)
- resize函数 (64)
- xftp6 (83)
- bt搜索 (75)
- c#var (76)
- mybatis大于等于 (64)
- xcode-select (66)
- mysql授权 (74)
- 下载测试 (70)
- linuxlink (65)
- pythonwget (67)
- androidinclude (65)
- libcrypto.so (74)
- logstashinput (65)
- hadoop端口 (65)
- vue阻止冒泡 (67)
- jquery跨域 (68)
- php写入文件 (73)
- kafkatools (66)
- mysql导出数据库 (66)
- jquery鼠标移入移出 (71)
- 取小数点后两位的函数 (73)