Java反射与动态代理:框架设计的基石
一、反射机制深度解剖(Java 17+新特性)
1. Class对象获取六大途径
// 1. 类名.class
Class<?> clazz1 = String.class; // 2. 对象.getClass()
String str = "";
Class<?> clazz2 = str.getClass(); // 3. Class.forName()
Class<?> clazz3 = Class.forName("java.lang.String"); // 4. 类加载器.loadClass()
Class<?> clazz4 = ClassLoader.getSystemClassLoader().loadClass("java.lang.String"); // 5. 基本类型TYPE字段
Class<?> clazz5 = int.TYPE; // 6. Java 9+模块反射(需opens授权)
Module module = clazz1.getModule();
2. 反射操作核心API
| 操作类型 | API示例 | 安全风险 |
|---|---|---|
| 字段访问 | Field.setAccessible(true) | 突破封装性 |
| 方法调用 | Method.invoke(obj, args) | 泛型擦除导致类型错误 |
| 构造对象 | Constructor.newInstance() | 绕过单例模式 |
| 注解扫描 | getDeclaredAnnotations() | 暴露敏感信息 |
二、动态代理两大实现方案对比
1. JDK动态代理(接口代理)
public class JdkProxyDemo { public static void main(String[] args) { Object target = new TargetImpl(); Object proxy = Proxy.newProxyInstance( target.getClass().getClassLoader(), target.getClass().getInterfaces(), (p, method, args) -> { System.out.println("Before method: " + method.getName()); return method.invoke(target, args); } ); ((TargetInterface) proxy).doSomething(); }
}
限制:必须基于接口,无法代理类
2. CGLIB字节码增强(类代理)
public class CglibProxyDemo { public static void main(String[] args) { Enhancer enhancer = new Enhancer(); enhancer.setSuperclass(TargetClass.class); enhancer.setCallback((MethodInterceptor) (obj, method, args, proxy) -> { System.out.println("Before method: " + method.getName()); return proxy.invokeSuper(obj, args); }); TargetClass proxy = (TargetClass) enhancer.create(); proxy.doSomething(); }
}
优势:支持类代理,性能接近原生调用
三、反射与代理在主流框架中的应用
1. Spring IOC容器实现原理
// Bean实例化核心逻辑(简化版)
BeanDefinition beanDefinition = getBeanDefinition(beanName);
Class<?> beanClass = beanDefinition.getBeanClass();
Constructor<?> constructor = beanClass.getDeclaredConstructor();
constructor.setAccessible(true);
Object bean = constructor.newInstance(); // 依赖注入
for (Field field : beanClass.getDeclaredFields()) { if (field.isAnnotationPresent(Autowired.class)) { Object dependency = getBean(field.getType()); field.setAccessible(true); field.set(bean, dependency); }
}
2. MyBatis Mapper接口动态代理
public class MapperProxy<T> implements InvocationHandler { private final SqlSession sqlSession; public Object invoke(Object proxy, Method method, Object[] args) { String methodName = method.getName(); String sqlId = method.getDeclaringClass().getName() + "." + methodName; return sqlSession.selectOne(sqlId, args[0]); } public static <T> T newInstance(Class<T> mapperInterface, SqlSession sqlSession) { return (T) Proxy.newProxyInstance( mapperInterface.getClassLoader(), new Class[]{mapperInterface}, new MapperProxy<>(sqlSession) ); }
}
四、高性能反射优化方案
1. 反射性能对比(纳秒级测试)
| 调用方式 | 首次调用 | 缓存后调用 |
|---|---|---|
| 直接调用 | 10 ns | 10 ns |
| 传统反射 | 2000 ns | 1500 ns |
| MethodHandle | 500 ns | 50 ns |
| Unsafe.allocateInstance | 30 ns | 30 ns |
2. 优化策略
- 缓存反射对象:复用
Method/Field实例 - 使用MethodHandle:
MethodHandles.Lookup lookup = MethodHandles.lookup(); MethodHandle mh = lookup.findVirtual(String.class, "length", MethodType.methodType(int.class)); int len = (int) mh.invokeExact("hello"); - LambdaMetafactory生成字节码(Java 8+)
- 禁用访问检查:
setAccessible(true)
五、安全攻防与漏洞防护
1. 反射攻击场景
- 单例模式破坏:
Constructor<?> constructor = Singleton.class.getDeclaredConstructor(); constructor.setAccessible(true); Singleton instance2 = (Singleton) constructor.newInstance(); - 注入恶意代码:通过反射修改
SecurityManager
2. 防御方案
- 模块化封装(Java 9+):
module my.module { opens com.example to spring.core; // 仅对指定模块开放反射 } - 安全管理器:
System.setSecurityManager(new SecurityManager() { @Override public void checkPackageAccess(String pkg) { if (pkg.startsWith("sun.misc")) throw new SecurityException(); } });
六、企业级实战案例
1. RPC框架动态代理实现
public class RpcProxy implements InvocationHandler { private final Class<?> serviceInterface; public Object invoke(Object proxy, Method method, Object[] args) { RpcRequest request = buildRequest(method, args); return transport.send(request).getResult(); } public static <T> T createProxy(Class<T> interfaceClass) { return (T) Proxy.newProxyInstance( interfaceClass.getClassLoader(), new Class[]{interfaceClass}, new RpcProxy(interfaceClass) ); }
}
2. 注解驱动AOP切面
@Aspect
public class LogAspect { @Around("@annotation(com.example.Log)") public Object around(ProceedingJoinPoint pjp) { Method method = ((MethodSignature) pjp.getSignature()).getMethod(); System.out.println("Enter method: " + method.getName()); return pjp.proceed(); }
}
七、跨技术对比(Java vs Python vs C#)
| 特性 | Java | Python | C# |
|---|---|---|---|
| 反射性能 | 中等(需优化) | 慢(动态类型) | 快(IL层支持) |
| 动态代理实现 | JDK Proxy/CGLIB | __getattr__魔法方法 | RealProxy类 |
| 元编程能力 | 有限(安全限制) | 灵活(猴子补丁) | 强(表达式树) |
| AOP框架支持 | Spring AOP/ AspectJ | Decorator/ wrapt | PostSharp |
八、高频面试题与避坑指南
1. 面试题:JDK动态代理为什么必须基于接口?
答案:
- Java单继承限制,代理类需继承Proxy类
- 兼容原有接口体系,保证类型安全
2. 常见陷阱:Lambda表达式导致的内存泄漏
// Method对象被Lambda持有导致无法回收
Method method = MyClass.class.getMethod("test");
Runnable r = () -> method.invoke(obj); // 持有method引用!
解决:改用MethodHandle或弱引用
相关文章:
Java反射与动态代理:框架设计的基石
一、反射机制深度解剖(Java 17新特性) 1. Class对象获取六大途径 // 1. 类名.class Class<?> clazz1 String.class; // 2. 对象.getClass() String str ""; Class<?> clazz2 str.getClass(); // 3. Class.forName(…...
day19-前端Web——Vue3+TS+ElementPlus
目录 1. Vue工程化1.1 介绍1.2 环境准备1.2.1 NodeJS安装双击安装包选择安装目录验证NodeJS环境变量配置npm的全局安装路径 1.3 Vue项目-创建1.4 Vue项目开发流程1.5 API风格1.6 案例 2. TS2.1 概述2.2 快速入门2.3 常用类型2.3.1 基础类型2.3.2 联合类型2.3.3 函数类型2.3.4 对…...
隐私保护在 Facebook 用户身份验证中的应用
在这个数字化的时代,个人隐私保护成为了公众关注的焦点。社交媒体巨头 Facebook 作为全球最大的社交平台之一,拥有数十亿用户,其在用户身份验证过程中对隐私保护的重视程度直接影响着用户的安全感和信任度。本文将探讨 Facebook 在用户身份验…...
【JavaWeb学习Day23】
Maven高级 分模块设计与开发 分模块设计:将一个大项目分成若干个子模块,方便项目的维护、扩展,也方便模块间的相互引用,资源共享。 策略: 1.策略一:按照功能模块拆分,比如:公共组…...
个人记录的一个插件,Unity-RuntimeMonitor
没有什么干货,仅仅是个人的记录 基于GUI做的一个工具:好处就是Monitor必须,Unity天然支持实时的Monitor;唯一不好处,就是默认字体太小了,layout居中,居右也是要自行设计的。 (下面文字是有一点点写错,但意思和功能就很牛逼了;并不是都按2 x shift,而是一个 shift 添…...
【NexLM 开源系列】如何封装多个大模型 API 调用
🌟 在这系列文章中,我们将一起探索如何搭建一个支持大模型集成项目 NexLM 的开发过程,从 架构设计 到 代码实战,逐步搭建一个支持 多种大模型(GPT-4、DeepSeek 等) 的 一站式大模型集成与管理平台ÿ…...
Git和GitHub基础教学
文章目录 1. 前言2. 历史3. 下载安装Git3.1 下载Git3.2 安装Git3.3 验证安装是否成功 4. 配置Git5. Git基础使用5.1 通过Git Bash使用5.1.1 创建一个新的仓库。5.1.1.1 克隆别人的仓库5.1.1.2 自己创建一个本地仓库 5.1.2 管理存档 5.2 通过Visual Studio Code使用 6. Git完成远…...
笔记六:单链表链表介绍与模拟实现
在他一生中,从来没有人能够像你们这样,以他的视角看待这个世界。 ---------《寻找天堂》 目录 文章目录 一、什么是链表? 二、为什么要使用链表? 三、 单链表介绍与使用 3.1 单链表 3.1.1 创建单链表节点 3.1.2 单链表的头插、…...
坐落于杭州的电商代运营公司品融电商
坐落于杭州的电商代运营公司品融电商 在中国电商行业蓬勃发展的浪潮中,品融电商(PINKROON)作为一家扎根杭州的新锐品牌管理公司,凭借其独特的全域增长方法论和实战经验,迅速崛起为行业标杆。自2020年成立以来&#x…...
微前端之 Garfish.js 的基础使用教程和进阶配置
前言 在现代前端开发中,微前端架构逐渐成为一种流行的解决方案。它允许将大型应用拆分成多个小型独立的子应用,从而提高开发效率和可维护性。Garfish.js 是一个强大的微前端框架,可以帮助我们轻松实现这一架构。在本文中,通过一个…...
图像的特征
图像的特征主要包括以下几类: 1. 颜色特征: 直方图:描述图像中颜色的分布。 颜色矩:通过颜色的均值、方差等统计量表示颜色分布。 主色调:图像中占主导地位的颜色。 2. 纹理特征: 灰度共生矩阵࿰…...
Spring上下文工具类
文章目录 获取ip地址请求上下文相关Spring上下文获取Bean对象 获取ip地址 public class IpUtils {private IpUtils() {}/*** 获取请求ip地址** return {link String}*/public static String getIpAddress() {HttpServletRequest request RequestContextHolderUtils.getReques…...
JSONUtil InvocationTargetException: null
使用JSONUtil时候报错,一般这时候你检查下自己代码里是不是重写了get或者set InvocationTargetException 是 Java 中的一个异常,通常在使用反射(Reflection)或动态代理(Dynamic Proxy)时抛出。它表示在调用…...
高压为什么cover不住低压的hold问题
常规下我们认为hold问题常发生在高压下,但很多情况下高压的hold无法cover低压的hold hold slack (tlaunch -tcapture) (tcqtcomb) -thold -tuncertainty (tlaunch -tcapture):代表时钟skew (tcqtcomb):代表data path的长度 thold:代表查表…...
【算法学习之路】8.栈和队列
栈和队列 前言一.简介二.题目12 前言 我会将一些常用的算法以及对应的题单给写完,形成一套完整的算法体系,以及大量的各个难度的题目,目前算法也写了几篇,题单正在更新,其他的也会陆陆续续的更新,希望大家点…...
OpenMCU(三):STM32F103 FreeRTOS移植
概述 本文主要描述了STM32F103移植FreeRTOS的简要步骤。移植描述过程中,忽略了Keil软件的部分使用技巧。默认读者熟练使用Keil软件。本文的描述是基于OpenMCU_RTOS这个工程,该工程已经下载放好了移植STM32F103 FreeRTOS的所有文件 OpenMCU_RTOS工程的愿景…...
大数据 spark hive 总结
Apache Spark 简介 是一个开源的统一分析引擎,专为大规模数据处理而设计。它提供了高级API,支持Java、Scala、Python和R语言,并且包含了一个优化过的执行引擎,该引擎支持循环计算(如机器学习算法)和交互式…...
小程序开发总结
今年第一次帮别人做小程序。 从开始动手到完成上线,一共耗时两天。AI 让写代码变得简单、高效。 不过,小程序和 Flutter 等大厂开发框架差距实在太大,导致我一开始根本找不到感觉。 第一,IDE 不好用,各种功能杂糅在…...
QLoggingCategory类使用
QLoggingCategory类使用 QLoggingCategory的概述 QLoggingCategory是Qt的日志策略类;可以通过声明不同的日志策略对象来输出不同的日志信息。打印信息类型如下:宏 Q_DECLARE_LOGGING_CATEGORY(name) 定义一个返回QLoggingCategory对象函数,…...
GPU加速生信分析-宏基因组MAG去污染
Deepurify利用多模态深度语言模型来过滤污染的基因组,从而提高了宏基因组组装基因组(MAGs)的质量,并且可以利用GPU加速。 宏基因组组装的基因组 (MAG) 为使用宏基因组测序数据探索微生物暗物质提供了有价值…...
数据结构(蓝桥杯常考点)
数据结构 前言:这个是针对于蓝桥杯竞赛常考的数据结构内容,基础算法比如高精度这些会在下期给大家总结 数据结构 竞赛中,时间复杂度不能超过10的7次方(1秒)到10的8次方(2秒) 空间限制&#x…...
从0到1入门Linux
一、常用命令 ls 列出目录内容 cd切换目录mkdir创建新目录rm删除文件或目录cp复制文件或目录mv移动或重命名文件和目录cat查看文件内容grep在文件中查找指定字符串ps查看当前进程状态top查看内存kill终止进程df -h查看磁盘空间存储情况iotop -o直接查看比较高的磁盘读写程序up…...
灰色地带规避:知识产权校验API的商标库模糊匹配算法
在反向海淘或其他电商业务场景中,为了规避知识产权方面的灰色地带,开发知识产权校验 API 并运用商标库模糊匹配算法是很有必要的。以下将详细介绍商标库模糊匹配算法的设计与实现: 算法设计思路 商标库模糊匹配算法的核心目标是在给定一个待匹…...
React:类组件(中)
dangerouslySetInnerHTML React写进{}内的东西,不允许被当作代码块解析,是为了防止xss攻击和代码注入 XSS(跨站脚本攻击,Cross-Site Scripting) 是一种常见的安全漏洞,攻击者通过注入恶意脚本到网页中&…...
第六次CCF-CSP认证(含C++源码)
第六次CCF-CSP认证 数位之和(easy)思路及AC代码遇到的问题 开心消消乐(easy)思路及AC代码 画图(mid)思路及AC代码 数位之和(easy) 题目链接 思路及AC代码 既然题目要求我们输出各位…...
SpringBoot 如何调用 WebService 接口
前言 调用WebService接口的方式有很多,今天记录一下,使用 Spring Web Services 调用 SOAP WebService接口 一.导入依赖 <!-- Spring Boot Web依赖 --><dependency><groupId>org.springframework.boot</groupId><artifactId…...
算法 之 树形dp 树的中心、重心
文章目录 重心实践题目小红的陡峭值 在树的算法中,求解树的中心和重心是一类十分重要的算法 求解树的重心 树的重心的定义:重心是树中的一个节点,如果将这个点删除后,剩余各个连通块中点数的最大值最小,那么这个节点…...
Docker 配置镜像源
》》Daemon {"registry-mirrors": ["https://docker.1ms.run","https://docker.xuanyuan.me"] }》》》然后在重新 docker systemctl restart docker...
Linux 离线部署Ollama和DeepSeek-r1模型
都在复制粘贴联网状态下linux部署deepseek,离线状态下需要下载Ollama和DeepSeek模型,然后将下载包上传到linux中。 1、下载Ollama https://github.com/ollama/ollama/releases 注意:如果CentOS7建议安装V0.5.11版本,V0.5.13需要…...
SQLAlchemy系列教程:如何执行原生SQL
Python中的数据库交互提供了高级API。但是,有时您可能需要执行原始SQL以提高效率或利用数据库特定的特性。本指南介绍在SQLAlchemy框架内执行原始SQL。 在SQLAlchemy中执行原生SQL SQLAlchemy虽然以其对象-关系映射(ORM)功能而闻名ÿ…...
