news 2026/6/15 18:34:57

Java反射

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Java反射

一、先搞懂:反射到底是个什么东西?

反射就是程序在运行的时候,能够动态获取类的所有信息(比如成员变量、方法、构造器),并且还能直接操作这些信息的一种机制

举个通俗的例子:平时我们写Java代码,都是先知道要用到哪个类,然后new一个对象,再调用它的方法——就像你知道要找小明借东西,直接走到他面前说“把东西借我用用”。而反射不一样,它是你不知道具体找哪个人,但能通过某种“侦查手段”找到这个人,还能强行用他的东西——这就是“动态”的意思,运行时才确定要操作的类和对象。

二、为什么要学反射?

刚开始我也疑惑:好好的直接调用方法不香吗?为啥非要搞反射这么复杂的东西?总结下来有3个核心作用:

  1. 实现动态创建对象和调用方法:这是最核心的作用。比如我们写一个通用的工具类,需要适配不同的类,这时候就不能把类名写死,而是通过反射动态获取类信息、创建对象。

  2. 突破类的封装性,操作私有成员:平时我们写的private成员变量、private方法,外部是没法直接访问的,但反射可以。比如有时候我们需要修改一个类的私有变量,又不想改这个类的源码(比如用别人写的jar包),这时候反射就派上用场了。

  3. 实现通用编程,提高代码复用性:比如写一个通用的对象拷贝工具、通用的数据库操作工具,不需要针对每个类单独写代码,而是通过反射获取类的成员变量,动态赋值和读取,大大减少重复代码。

这里要提醒一句:反射虽然强大,但也不能随便用,因为它打破了封装性,会让代码的安全性降低,而且运行效率比直接调用要低一点,平时开发中如果能直接调用,就别用反射~

三、新手入门:反射的核心API实操(附代码示例)

理论讲再多不如写一遍代码,反射的核心操作其实就围绕3个步骤:

获取Class对象 → 通过Class对象获取类的内部信息(成员变量、方法、构造器) → 操作这些内部信息(创建对象、调用方法、修改变量)

下面用一个简单的Student类来演示

第一步:先定义一个测试用的Student类

public class Student { // 成员变量(包含public和private) public String name; private int age; // 构造器(无参和有参) public Student() { } public Student(String name, int age) { this.name = name; this.age = age; } // 成员方法(包含public和private) public void study() { System.out.println(name + "正在学习Java反射~"); } private void eat(String food) { System.out.println(name + "正在吃" + food); } @Override public String toString() { return "Student{name='" + name + "', age=" + age + "}"; } }

第二步:核心操作1:获取Class对象

要使用反射,第一步必须获取目标类的Class对象,这是反射的入口:

public class ReflectionDemo { public static void main(String[] args) throws ClassNotFoundException { // 方式1:通过类名.class获取(最常用,编译时就确定) Class<?> clazz1 = Student.class; // 方式2:通过对象.getClass()获取(需要先创建对象,适合已经有实例的情况) Student student = new Student(); Class<?> clazz2 = student.getClass(); // 方式3:通过Class.forName("全类名")获取(动态加载,最灵活,运行时确定) // 注意:全类名是包名+类名,比如我的Student类在com.test包下 Class<?> clazz3 = Class.forName("com.test.Student"); // 验证一下:这三个Class对象是同一个(一个类只有一个Class对象) System.out.println(clazz1 == clazz2); // true System.out.println(clazz1 == clazz3); // true } }

这里要注意:Class.forName()方法需要处理ClassNotFoundException异常,要么throws要么try-catch

第三步:核心操作2:通过Class对象操作类的内部信息

获取到Class对象后,就可以通过它的API获取成员变量、方法、构造器,然后进行操作了。下面分场景演示几个常用的操作:

场景1:通过反射创建对象(两种方式)
public class ReflectionDemo { public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchMethodException, InvocationTargetException { Class<?> clazz = Class.forName("com.test.Student"); // 方式1:通过无参构造器创建对象(需要类有public的无参构造器) Object student1 = clazz.newInstance(); System.out.println(student1); // 输出:Student{name='null', age=0} // 方式2:通过有参构造器创建对象 // 1. 先获取有参构造器:参数是构造器的参数类型.class Constructor<?> constructor = clazz.getConstructor(String.class, int.class); // 2. 调用构造器的newInstance()方法创建对象,传入实际参数 Object student2 = constructor.newInstance("小明", 20); System.out.println(student2); // 输出:Student{name='小明', age=20} } }
场景2:通过反射获取和修改成员变量(包括private)
public class ReflectionDemo { public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchFieldException { Class<?> clazz = Class.forName("com.test.Student"); Object student = clazz.newInstance(); // 1. 获取public成员变量name并修改 Field nameField = clazz.getField("name"); nameField.set(student, "小红"); // 给student对象的name属性赋值 System.out.println(nameField.get(student)); // 获取name属性的值,输出:小红 // 2. 获取private成员变量age并修改(需要先设置setAccessible(true)打破封装) Field ageField = clazz.getDeclaredField("age"); ageField.setAccessible(true); // 关键:允许访问private成员 ageField.set(student, 19); // 给private的age属性赋值 System.out.println(ageField.get(student)); // 获取age属性的值,输出:19 } }

这里重点说一下:getField()只能获取public的成员变量,要获取private的必须用getDeclaredField(),并且要调用setAccessible(true)来关闭Java的访问检查,这样才能操作private成员。

场景3:通过反射调用成员方法(包括private)
public class ReflectionDemo { public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchMethodException, InvocationTargetException { Class<?> clazz = Class.forName("com.test.Student"); Object student = clazz.newInstance(); // 1. 先给name属性赋值,方便方法输出 Field nameField = clazz.getField("name"); nameField.set(student, "小李"); // 2. 调用public方法study() Method studyMethod = clazz.getMethod("study"); studyMethod.invoke(student); // 调用方法,输出:小李正在学习Java反射~ // 3. 调用private方法eat(String food) Method eatMethod = clazz.getDeclaredMethod("eat", String.class); // 第二个参数是方法的参数类型.class eatMethod.setAccessible(true); // 允许访问private方法 eatMethod.invoke(student, "汉堡"); // 调用方法,传入参数,输出:小李正在吃汉堡 } }

这里要注意:

getMethod()获取public方法,getDeclaredMethod()获取所有方法(包括private);

调用方法时用invoke(),第一个参数是要调用方法的对象,后面的参数是方法的实际参数。

四、需要注意的点

  1. 忘记处理异常:反射的大部分API都会抛出checked异常(比如ClassNotFoundException、NoSuchMethodException),新手容易忘写try-catch或者throws,导致编译报错。

  2. 用getField()获取private变量:getField()只能获取public的,获取private的必须用getDeclaredField(),还要加setAccessible(true)。

  3. 获取有参构造器时参数类型写错:比如构造器是Student(String name, int age),获取时写成getConstructor(String.class, String.class),就会报NoSuchMethodException。

  4. 调用invoke()时忘记传对象:非静态方法必须传入要调用的对象,静态方法可以传null。

  5. 认为反射能修改final变量:虽然反射能获取final变量,但修改它的结果是不确定的,不同JVM可能有不同表现,尽量不要这么做。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/15 16:48:51

仅限早期用户掌握的技巧:利用Open-AutoGLM 2.0实现毫秒级响应推理部署

第一章&#xff1a;Open-AutoGLM 2.0 毫秒级推理的背景与意义随着大语言模型在自然语言处理领域的广泛应用&#xff0c;推理效率成为制约其落地的关键瓶颈。传统模型在响应延迟、资源消耗和部署成本方面难以满足实时交互场景的需求&#xff0c;尤其是在智能客服、边缘计算和移动…

作者头像 李华
网站建设 2026/6/15 17:59:12

类脑智能技术与系统——脉冲神经网络(下)

第三节&#xff1a;代理梯度法及ANN-to-SNN转换一、代理梯度&#xff08;Surrogate Gradient&#xff09;法&#xff08;一&#xff09;核心问题&#xff1a;为什么需要代理梯度&#xff1f;1. 代理梯度怎么做脉冲神经元的激活函数&#xff08;发放脉冲&#xff09;是一个不可微…

作者头像 李华
网站建设 2026/6/15 13:09:40

H3C路由策略配置方法及命令

H3C路由策略配置文档&#xff08;含热门设备案例&#xff09;一、概述1.1 路由策略定义路由策略是网络设备通过一系列规则对路由信息的生成、发布、接收和转发进行控制的技术&#xff0c;核心目的是优化网络路由选路、实现流量引导、保障网络安全与稳定。H3C设备的路由策略主要…

作者头像 李华
网站建设 2026/6/15 14:03:13

结合Playwright抓取动态网页内容注入知识库

结合Playwright抓取动态网页内容注入知识库 在构建企业级智能问答系统时&#xff0c;一个常被忽视但至关重要的问题浮出水面&#xff1a;我们如何让AI真正“读懂”那些由JavaScript驱动的现代网页&#xff1f; 传统爬虫通过requests获取HTML源码的方式&#xff0c;在面对单页应…

作者头像 李华
网站建设 2026/6/15 15:11:21

基于深度学习的消防早期火源探测系统的设计与实现任务书

* 课题的内容和要求&#xff1a;&#xff08;一&#xff09;内容&#xff1a;选题背景与意义&#xff1a;随着城市化进程的加速&#xff0c;消防安全问题日益凸显&#xff0c;尤其是火灾的早期发现和及时应对对于减少人员伤亡和财产损失至关重要。传统的消防监控系统往往依赖于…

作者头像 李华
网站建设 2026/6/15 15:20:18

Spring AOP是啥?和动态代理啥关系?项目中怎么用?

在Java企业级开发中&#xff0c;Spring框架的AOP&#xff08;面向切面编程&#xff09;是一种至关重要的编程范式。它并非用于构建核心业务逻辑&#xff0c;而是提供了一种优雅的方式来处理那些散布在应用多个模块中的“横切关注点”&#xff0c;例如日志记录、事务管理和安全检…

作者头像 李华