Java进阶——反射机制超全详解

        反射能在运行时动态操作类和对象的能力,极大地增强了程序的灵活性。但是反射的使用也伴随着性能开销和安全风险。本文将由博主带你一起深入探讨 Java 反射的核心概念、关键类和方法,以及在日常开发中的应用场景。

本文目录

    • 一、反射的核心概念
      • 1. 运行时类型信息
      • 2. 核心类
    • 二、获取 Class 对象的四种方式
      • 1. 类名.class
      • 2. 对象.getClass()
      • 3. Class.forName()
      • 4. 类加载器获取
    • 三、反射操作字段
      • 1. 获取字段
      • 2. 访问私有字段
      • 3. 修改字段值
    • 四、反射调用方法
      • 1. 获取方法
      • 2. 方法调用
      • 3. 静态方法调用
    • 五、反射操作构造方法
      • 1. 创建对象实例
      • 2. 突破单例模式
    • 六、反射性能优化
      • 1. 缓存反射对象
      • 2. 避免频繁反射调用
    • 七、反射在日常工作中的实际应用场景
      • 1. 动态加载插件
      • 2. 单元测试工具
    • 总结

一、反射的核心概念

1. 运行时类型信息

反射允许在运行时获取类的完整结构,包括类名、方法、字段、注解等,并动态操作对象,突破编译时的类型限制。

2. 核心类

  • Class<T>:类的元数据入口,是所有反射操作的起点。
  • Field:代表类的字段信息,可用于获取和设置字段的值。
  • Method:代表类的方法信息,可用于调用类的方法。
  • Constructor<T>:代表类的构造方法信息,可用于创建对象实例。
  • Modifier:用于解析访问修饰符,如 publicprivate 等。


二、获取 Class 对象的四种方式

1. 类名.class

适用于编译时已知类的情况。

Class<User> clazz = User.class;

2. 对象.getClass()

通过对象实例获取其对应的 Class 对象。

User user = new User();
Class<?> clazz = user.getClass();

3. Class.forName()

通过类的全限定名动态加载类,需要处理 ClassNotFoundException

Class<?> clazz = Class.forName("com.shop.ecommerce.User");

4. 类加载器获取

在模块化环境中,可使用类加载器加载类。

ClassLoader loader = Thread.currentThread().getContextClassLoader();
Class<?> clazz = loader.loadClass("com.shop.ecommerce.User");



三、反射操作字段

1. 获取字段

  • getField(String name):仅获取公共字段(包括继承的)。
  • getDeclaredField(String name):获取本类声明的所有字段(包括 private)。

2. 访问私有字段

Field field = clazz.getDeclaredField("字段名");
field.setAccessible(true); //  设置是否可以访问 private
String value = (String) field.get(user);

3. 修改字段值

field.set(user, "newValue"); 



四、反射调用方法

1. 获取方法

  • getMethod(String name, Class<?>... paramTypes):获取公共方法(含继承)。
  • getDeclaredMethod(...):获取本类声明的所有方法。
// 获取 User 类的私有方法
Method method = clazz.getDeclaredMethod("方法名", int.class);
method.setAccessible(true);

2. 方法调用

Object result = method.invoke(user, 100);  // 参数需严格匹配

3. 静态方法调用

Method staticMethod = clazz.getMethod("getDefaultRole");
String role = (String) staticMethod.invoke(null); // 实例参数传 null



五、反射操作构造方法

1. 创建对象实例

Constructor<User> constructor = clazz.getConstructor(String.class, String.class);
User user = constructor.newInstance("user123", "VIP");

2. 突破单例模式

// 反射调用私有构造方法
Constructor<Singleton> constructor = Singleton.class.getDeclaredConstructor();
constructor.setAccessible(true);
Singleton instance = constructor.newInstance();



六、反射性能优化

1. 缓存反射对象

private static final Method calculateDiscountMethod;
static {try {calculateDiscountMethod = User.class.getDeclaredMethod("calculateDiscount", int.class);calculateDiscountMethod.setAccessible(true);} catch (NoSuchMethodException e) {throw new RuntimeException(e);}
}

2. 避免频繁反射调用

反射比直接调用慢 10 - 100 倍,应尽量减少反射调用的频率。



七、反射在日常工作中的实际应用场景

1. 动态加载插件

// 加载支付插件
Class<?> paymentClass = Class.forName("com.shop.payment.AlipayPlugin");
PaymentPlugin plugin = (PaymentPlugin) paymentClass.newInstance();
plugin.process(order);

2. 单元测试工具

// 测试私有方法
public class OrderServiceTest {@Testvoid testPrivateMethod() throws Exception {Method method = OrderService.class.getDeclaredMethod("validateStock", Long.class);method.setAccessible(true);boolean result = (boolean) method.invoke(orderService, 1001L);assertTrue(result);}
}



总结

        合理使用反射可以实现插件化架构、灵活的数据处理。



← 上一篇 Java进阶——Stream流以及常用方法详解
记得点赞、关注、收藏哦!
下一篇 Java进阶——注解一文全懂 →

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/896659.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

OAK相机的抗震性测试

在工业环境中&#xff0c;双目视觉相机必须具备与工作环境同等的坚固性。鉴于部分客户会将我们的相机应用于恶劣环境&#xff08;例如安装在重型机械上&#xff09;&#xff0c;我们依据EN 60068-2-6:2008标准对相机进行了振动耐受性测试。 测试涉及的相机型号包括&#xff1a…

Express MVC

1. 安装依赖 npm init -y npm install express npm install --save-dev typescript ts-node ejs types/node types/express tsc --init 2. 项目目录结构如下&#xff0c;没有的手动创建 /my-app/src/modelsuser.ts/viewsindex.ejsuserList.ejs/controllersuserController.ts…

apache-maven-3.2.1

MAVEN_HOME D:\apache-maven-3.2.1 PATH D:\apache-maven-3.2.1\bin cmd mvn -v <localRepository>d:\localRepository</localRepository> setting.xml <?xml version"1.0" encoding"UTF-8"?><!-- Licensed to the Apache Soft…

合并两个有序链表:递归与迭代的实现分析

合并两个有序链表&#xff1a;递归与迭代的实现分析 在算法与数据结构的世界里&#xff0c;链表作为一种基本的数据结构&#xff0c;经常被用来解决各种问题。特别是对于有序链表的合并&#xff0c;既是经典面试题&#xff0c;也是提高编程能力的重要练习之一。合并两个有序链…

破解密码防线:渗透测试中的密码攻击手法汇总

密码是网络安全中的一道重要防线&#xff0c;然而&#xff0c;若密码策略不严密&#xff0c;往往会为攻击者提供可乘之机。本文将简要介绍渗透测试中关于密码的几种常见攻击思路和手法。 1. 确认使用默认及常见的账号密码 在渗透测试的初期&#xff0c;攻击者通常会尝试使用系…

CSS Selectors

当然&#xff0c;理解纯CSS选择器&#xff08;CSS Selectors&#xff09;对于进行UI自动化测试非常重要。CSS选择器允许您通过元素的属性、层级关系、类名、ID等来精准定位页面上的元素。下面我将详细讲解CSS选择器的常见用法&#xff0c;并结合您的需求提供具体的示例。 1. 基…

【java】@Transactional导致@DS注解切换数据源失效

最近业务中出现了多商户多租户的逻辑&#xff0c;所以需要分库&#xff0c;项目框架使用了mybatisplus所以我们自然而然的选择了同是baomidou开发的dynamic.datasource来实现多数据源的切换。在使用初期程序运行都很好&#xff0c;但之后发现在调用com.baomidou.mybatisplus.ex…

浅入浅出Selenium DevTools

前言 在自动化测试领域&#xff0c;Selenium一直是主流工具之一。随着前端技术的不断发展&#xff0c;浏览器的功能也在不断丰富。 Selenium 3版本前&#xff0c;一套通用的采集流程如上图所示&#xff1a; 打开Charles&#xff0c;设置Session自动导出频次及导出路径Seleniu…

04 路由表的IP分组传输过程

目录 1、路由表的核心结构 2、IP分组传输过程和数据包转发过程 2.1、IP分组传输过程 2.2、数据包转发过程 2.3、IP分组传输过程和数据包转发的区别 3、数据包的变化 3.1、拓扑结构 3.2、传输过程详解&#xff08;主机A → 主机B&#xff09; 3.2.1、主机A发送数据 3.2…

【子网掩码计算器:Python + Tkinter 实现】

子网掩码计算器&#xff1a;Python Tkinter 实现 引言代码功能概述代码实现思路1. 界面设计2. 功能实现3. 事件处理 子网掩码计算器实现步骤1. 导入必要的库2. 定义主窗口类 SubnetCalculatorApp3. 创建菜单栏4. 创建界面组件5. 判断 IP 地址类别6. 计算子网信息7. 其他功能函…

【练习】【贪心】力扣1005. K 次取反后最大化的数组和

题目 1005 K 次取反后最大化的数组和 给你一个整数数组 nums 和一个整数 k &#xff0c;按以下方法修改该数组&#xff1a; 选择某个下标 i 并将 nums[i] 替换为 -nums[i] 。 重复这个过程恰好 k 次。可以多次选择同一个下标 i 。 以这种方式修改数组后&#xff0c;返回数组 可…

3dsmax中使用python创建PBR材质并挂接贴图

前言 笔者处理模型时下载到一个pbr材质库贴图包&#xff0c;手动每次创建材质过于麻烦&#xff0c;因此计划使用自动化脚本根据贴图名自动创建材质。 3dsmax的原本脚本使用的是maxscript&#xff0c;语法有点奇怪懒得学&#xff0c;发现也支持使用python编写脚本&#…

Metal学习笔记九:光照基础

光和阴影是使场景流行的重要要求。通过一些着色器艺术&#xff0c;您可以突出重要的对象、描述天气和一天中的时间并设置场景的气氛。即使您的场景由卡通对象组成&#xff0c;如果您没有正确地照亮它们&#xff0c;场景也会变得平淡无奇。 最简单的光照方法之一是 Phong 反射模…

JAVA学习笔记038——bean的概念和常见注解标注

什么是bean? Bean 就是 被 Spring 管理的对象&#xff0c;就像工厂流水线上生产的“标准产品”。这些对象不是你自己 new 出来的&#xff0c;而是由 Spring 容器&#xff08;一个超级工厂&#xff09;帮你创建、组装、管理。 由 Component、Service、Controller 等注解标记的…

start DL from stratch (2)!!!

start DL from stratch &#xff08;2&#xff09;!!! 一、CPU and GPUcpuGPU安培架构爱达洛夫莱斯架构 二、使用conda创建一个新的虚拟环境三、autodl操作先知Linux复习目录文件和数据上传对于整个镜像的操作守护进程Tips 四、autodl租用创建实例<big>没有所需要的版本的…

机器学习:线性回归,梯度下降

线性回归模型 (Linear Regression Model) 梯度下降算法 (Gradient Descent Algorithm) 的数学公式

论文笔记-NeurIPS2017-DropoutNet

论文笔记-NeurIPS2017-DropoutNet: Addressing Cold Start in Recommender Systems DropoutNet&#xff1a;解决推荐系统中的冷启动问题摘要1.引言2.前言3.方法3.1模型架构3.2冷启动训练3.3推荐 4.实验4.1实验设置4.2在CiteULike上的实验结果4.2.1 Dropout率的影响4.2.2 实验结…

nvm的学习

学习 nvm&#xff08;Node Version Manager&#xff09; 是掌握 Node.js 开发的关键技能之一。以下是系统的学习路径和实战指南&#xff0c;涵盖从基础到进阶的内容&#xff1a; 一、基础入门 1. nvm 的核心作用 多版本共存&#xff1a;安装和管理多个 Node.js 版本&#xff…

GPT-4.5实际性能评测:实际探索

摘要 经过数万轮严格测试&#xff0c;GPT-4.5的性能并未超越其前代产品GPT-4。此前发布的《GPT-4.5 一手实测&#xff1a;垃圾》一文中存在不准确描述&#xff0c;在此向读者致歉。尽管GPT-4.5在价格上有所提升且响应速度较慢&#xff0c;但测试结果显示其模型素质并未达到预期…

从UNIX到Linux:操作系统进化史与开源革命

从UNIX到Linux&#xff1a;操作系统进化史与开源革命 一、操作系统&#xff1a;数字世界的基石 1.1 什么是操作系统&#xff1f; 操作系统&#xff08;OS&#xff09;是计算机系统的核心管理者&#xff0c;承担着三大核心使命&#xff1a; 硬件指挥官&#xff1a;直接管理C…