当前位置: 首页 > news >正文

Java中的反射

目录

什么是反射

反射的核心作用

反射的核心类

反射的基本使用

获取Class对象

创建对象

操作字段(Field)

调用方法(Method)

反射的应用场景

反射的优缺点

优点

缺点

示例:完整反射操作

总结


什么是反射

能够获取类信息的能力叫做反射


反射的核心作用

反射的核心是 在运行时动态分析类或对象,而不是在编译时确定。主要功能包括:

  • 动态加载类Class.forName("全限定类名")

  • 获取类的结构信息(方法、字段、构造器、注解等)

  • 操作对象的属性和方法(包括私有成员)

  • 动态创建对象(即使构造器是 private 的)

  • 动态调用方法(包括私有方法)

反射的核心类

Java 反射主要通过以下类实现:

类/接口作用
java.lang.Class表示一个类或接口,是所有反射操作的入口
java.lang.reflect.Field表示类的字段(成员变量),可获取或修改字段值
java.lang.reflect.Method表示类的方法,可动态调用方法
java.lang.reflect.Constructor表示类的构造方法,可动态创建对象
java.lang.reflect.Modifier解析类、方法、字段的修饰符(如 publicprivatestatic 等)

画图理解反射


反射的基本使用

获取Class对象

要使用反射,必须先获取类的 Class 对象。有 3 种方式:

// 方式1:类名.class
Class<Cat> clazz1 = Cat.class;// 方式2:对象.getClass()
Cat cat = new Cat();
Class<?> clazz2 = cat.getClass();// 方式3:Class.forName("全限定类名")(最常用)
Class<?> clazz3 = Class.forName("com.qcby.reflect.Cat");

创建对象

通过 Constructor 动态创建对象:

// 获取无参构造器(即使是 private 的也可以访问)
Constructor<?> constructor = clazz.getDeclaredConstructor();
constructor.setAccessible(true); // 暴力反射,绕过 private 限制
Cat cat = (Cat) constructor.newInstance();// 获取有参构造器
Constructor<?> constructor2 = clazz.getDeclaredConstructor(int.class, String.class);
Cat cat2 = (Cat) constructor2.newInstance(3, "Tom");

操作字段(Field)

获取和修改对象的字段值(包括私有字段):

Field ageField = clazz.getDeclaredField("age");
ageField.setAccessible(true); // 访问私有字段// 获取字段值
int age = (int) ageField.get(cat);
System.out.println(age); // 输出:3// 修改字段值
ageField.set(cat, 5); // 把 cat 的 age 改为 5

调用方法(Method)

动态调用方法(包括私有方法):

Method runMethod = clazz.getDeclaredMethod("run", String.class);
runMethod.setAccessible(true); // 访问私有方法// 调用方法(相当于 cat.run("Jerry"))
runMethod.invoke(cat, "Jerry"); // 输出:Tom的猫正在追Jerry

反射的应用场景

  1. 框架开发(如 Spring 的依赖注入、Hibernate 的 ORM 映射)

  2. 动态代理(如 JDK 动态代理)

  3. 注解处理(如 @Autowired@RequestMapping

  4. 单元测试(Mock 对象)

  5. 动态加载类(如插件化开发)

反射的优缺点

优点

  • 灵活性高:可以在运行时动态操作类,实现高度解耦。

  • 突破访问限制:可以访问 private 成员,实现特殊需求(如序列化、反序列化)。

缺点

  • 性能较低:反射比直接调用方法慢(JVM 无法优化反射调用)。

  • 安全性问题:可以绕过访问控制,破坏封装性。

  • 代码可读性差:反射代码通常较难理解和维护。


示例:完整反射操作

package com.qcby.reflect;public class Cat implements Jump,Run{private int age;public String name;protected String color;double height;public Cat(int age, String name, String color, double height) {this.age = age;this.name = name;this.color = color;this.height = height;}private Cat( ) {}Cat(String color){this.color = color;}public void run(String name) {System.out.println("小猫的名字叫"+name);}private int setAge(int age) {System.out.println("aa"+age);return age;}void flay() {System.out.println("猫不会飞");}
}package com.qcby.reflect;import java.lang.reflect.Field;
import java.util.Arrays;public class Test {public static void main(String[] args) throws Exception {// 反射获取类信息// 1. 使用正确的类名路径Class clazz = Class.forName("com.qcby.reflect.Cat");// 2. 获取类信息// 获取所有声明的字段(包括私有字段)Field[] fields = clazz.getDeclaredFields();System.out.println(Arrays.toString(fields));// 获取单个字段(使用getDeclaredField,返回Field对象)Field ageField = clazz.getDeclaredField("age");System.out.println(ageField);Field nameField = clazz.getDeclaredField("name");System.out.println(nameField);Field colorField = clazz.getDeclaredField("color");System.out.println(colorField);Field heightField = clazz.getDeclaredField("height");System.out.println(heightField);// 获取所有公共字段(仅public)Field[] fields1 = clazz.getFields();System.out.println(Arrays.toString(fields1));// 获取单个公共字段(仅public)Field nameField1 = clazz.getField("name");System.out.println(nameField1);}
}package com.qcby.reflect;import java.lang.reflect.Constructor;
import java.util.Arrays;public class Test1 {public static void main(String[] args) throws Exception {Class clazz = Class.forName("com.qcby.reflect.Cat"); // 获取所有公共构造方法Constructor[] constructors = clazz.getConstructors();System.out.println(Arrays.toString(constructors));// 获取所有声明的构造方法(包括私有)Constructor[] declaredConstructors = clazz.getDeclaredConstructors();System.out.println(Arrays.toString(declaredConstructors));// 获取无参构造方法(需处理访问权限)Constructor constructor1 = clazz.getDeclaredConstructor();System.out.println(constructor1);// 获取带参数的构造方法Constructor constructor2 = clazz.getDeclaredConstructor(String.class);System.out.println(constructor2);// 获取公共无参构造方法Constructor constructor3 = clazz.getConstructor(int.class, String.class, String.class, double.class);System.out.println(constructor3);}
}package com.qcby.reflect;import java.lang.reflect.Method;
import java.util.Arrays;public class Test2 {public static void main(String[] args) throws Exception {Class clazz = Class.forName("com.qcby.reflect.Cat");// 获取所有声明的方法(包括私有方法)Method[] methods = clazz.getDeclaredMethods();System.out.println(Arrays.toString(methods));// 获取所有公共方法(包括继承的公共方法)Method[] methods2 = clazz.getMethods();System.out.println(Arrays.toString(methods2));// 根据方法名和参数类型获取方法Method runMethod1 = clazz.getDeclaredMethod("run", String.class);System.out.println(runMethod1);Method runMethod2 = clazz.getMethod("run", String.class);System.out.println(runMethod2);Method flyMethod1 = clazz.getDeclaredMethod("flay");System.out.println(flyMethod1);}
}

总结

特性说明
动态性运行时获取类信息,无需提前知道类的结构
灵活性可以操作私有成员、动态创建对象、调用方法
用途广泛框架开发、动态代理、注解处理、单元测试等
性能损耗比直接调用满,不适合高频场景
安全性可以绕过访问控制,需谨慎使用

反射是 Java 高级特性,合理使用可以极大增强程序的灵活性,但滥用会导致性能问题和代码难以维护。建议在框架开发、动态代理等场景使用,普通业务代码尽量避免。

相关文章:

  • T-SQL在SQL Server中判断表、字段、索引、视图、触发器、Synonym等是否存在
  • MCP协议将颠覆传统数据集成
  • 2025-05-09 提示学习概念
  • QML AnimatedImage组件详解
  • 【动手学大模型开发 20】使用 Streamlit 部署大模型 RAG应用
  • C++跨平台开发实践:深入解析与常见问题处理指南
  • 西门子PLC串口转网口模块:工业通信的智能桥梁
  • 25FIC初赛(介质)
  • BUUCTF——杂项渗透之赛博朋克
  • 视频流:大华及海康视频流本地测试预览
  • 单调栈所有模版类型(4)
  • Windows11下通过Docker安装Redis
  • A2A与MCP定义下,User,Agent,api(tool)间的交互流程图
  • 工业相机的作用及未来发展
  • spring ai alibaba ChatClient 获取大模型返回内容的方式 以及使用场景
  • 多分类问题softmax传递函数+交叉熵损失
  • [特征工程]机器学习-part2
  • 物流基础知识-术语 | 医药物流(1)
  • 操作系统面试题(3)
  • 北斗导航 | RTKLib中重难点技术,公式,代码
  • 上海发布首份直播电商行业自律公约,禁止虚假宣传、商业诋毁
  • 央行:当前我国债券市场定价效率、机构债券投资交易和风险管理能力仍有待提升
  • 聆听百年唐调正声:唐文治王蘧常吟诵传习的背后
  • 深圳两家会所涉卖淫嫖娼各被罚7万元逾期未缴,警方发催告书
  • 浙江一民企拍地后遭政府两次违约,“民告官”三年又提起民事诉讼
  • 欧盟委员会计划对950亿欧元美国进口产品采取反制措施