news 2026/4/22 17:33:07

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/4/9 19:52:18

【完整源码+数据集+部署教程】鹰鳐检测系统源码分享[一条龙教学YOLOV8标注好的数据集一键训练_70+全套改进创新点发刊_Web前端展示]

一、背景意义 随着生态环境的变化和人类活动的加剧&#xff0c;海洋生物的保护已成为全球关注的焦点之一。鹰鳐&#xff08;Eagle Ray&#xff09;作为一种重要的海洋生物&#xff0c;其独特的生态地位和美丽的外形使其成为研究海洋生态系统的重要对象。然而&#xff0c;鹰鳐的…

作者头像 李华
网站建设 2026/4/21 3:44:56

从“经验驱动”到“数据驱动”:数值赋能下轨道交通安全管理工作的范式重构与实践路径

目录 一、引言&#xff1a;安全管理的时代挑战与范式革新机遇 二、数值赋能安全管理的内涵与理论框架 三、数值赋能的四大核心应用场景 四、面临的深层挑战 五、从技术赋能到体系重构&#xff1a;实现路径建议 六、结论与展望 摘要&#xff1a; 安全是城市轨道交通运营的…

作者头像 李华
网站建设 2026/4/22 16:08:26

50个人的企业网络组建技术方案

50人的公司组网方案参考一、方案概述本方案针对50人办公室的网络需求设计&#xff0c;旨在构建一套稳定可靠、高速高效、易于管理且具备可扩展性的局域网系统。方案充分考虑日常办公&#xff08;如文件传输、邮件收发、网页浏览&#xff09;、多媒体会议、云服务访问等核心场景…

作者头像 李华
网站建设 2026/4/23 13:10:44

从安装到推理全流程:Open-AutoGLM笔记本部署实录,新手必看

第一章&#xff1a;Open-AutoGLM部署概述Open-AutoGLM 是一个开源的自动化通用语言模型推理框架&#xff0c;专为简化大语言模型在本地及边缘设备上的部署流程而设计。它支持多种后端推理引擎&#xff08;如 ONNX Runtime、TensorRT 和 llama.cpp&#xff09;&#xff0c;并提供…

作者头像 李华
网站建设 2026/4/7 2:36:25

从测试到生产:Open-AutoGLM服务器部署完整路径(含高可用架构设计)

第一章&#xff1a;从测试到生产的Open-AutoGLM部署概述在构建现代化的生成式AI应用时&#xff0c;Open-AutoGLM作为一款支持自动化语言理解与生成任务的开源框架&#xff0c;其从测试环境到生产环境的部署流程至关重要。该过程不仅涉及模型性能验证&#xff0c;还包括服务稳定…

作者头像 李华