news 2026/3/17 8:40:23

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/3/12 7:34:26

【Win系统部署Open-AutoGLM全攻略】:手把手教你5步完成本地大模型部署

第一章&#xff1a;Win系统部署Open-AutoGLM概述在Windows操作系统上部署Open-AutoGLM&#xff0c;为本地化大模型推理与自动化任务执行提供了高效支持。该框架结合了AutoGLM的自然语言理解能力与轻量化部署特性&#xff0c;适用于企业内部知识库、智能客服及自动化办公场景。环…

作者头像 李华
网站建设 2026/3/13 19:04:43

Open-AutoGLM接口兼容性问题全攻克,资深架构师总结的8条黄金法则

第一章&#xff1a;Open-AutoGLM跨应用操作Open-AutoGLM 是一个支持多应用协同的自动化语言模型集成框架&#xff0c;能够在异构系统间实现指令解析与任务调度。其核心能力在于通过统一接口调用不同应用程序的API&#xff0c;并基于自然语言输入生成可执行的操作序列。配置跨应…

作者头像 李华
网站建设 2026/3/14 14:44:40

Open-AutoGLM部署避坑指南:7个关键步骤让你一次成功

第一章&#xff1a;Open-AutoGLM部署概述Open-AutoGLM 是一个开源的自动化通用语言模型&#xff08;GLM&#xff09;部署框架&#xff0c;旨在简化大语言模型在生产环境中的集成与运维。该框架支持多种硬件平台和推理后端&#xff0c;提供模块化配置、自动扩缩容以及API网关集成…

作者头像 李华