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

动态代理选择:JDK vs CGLIB

在Java开发中,动态代理是一种强大的机制,它允许我们在运行时创建代理对象,从而在不修改原有代码的情况下,对目标对象的方法进行增强。这种技术在AOP(面向切面编程)、RPC(远程过程调用)、ORM(对象关系映射)等领域有着广泛的应用。Java中实现动态代理主要有两种方式:JDK动态代理和CGLIB动态代理。

一、JDK动态代理

1. 原理

JDK动态代理是Java语言自带的代理实现方式,它基于Java的反射机制。当使用JDK动态代理时,代理类会实现目标对象所实现的接口,并在运行时动态生成。其核心在于java.lang.reflect.Proxy类和java.lang.reflect.InvocationHandler接口。Proxy类负责创建代理对象,而InvocationHandler接口则定义了代理对象的方法调用逻辑。当代理对象的方法被调用时,实际会转发到InvocationHandlerinvoke方法,我们可以在invoke方法中对目标方法进行增强。

2. 特点

  • 基于接口: JDK动态代理只能代理实现了接口的类。如果目标对象没有实现任何接口,则无法使用JDK动态代理。
  • 运行时生成: 代理类在运行时动态生成,并加载到JVM中。
  • 性能: 在Java 8及以前的版本中,JDK动态代理的性能通常低于CGLIB。但在Java 9及以后的版本中,由于JVM对动态代理的优化,其性能已有所提升,甚至在某些场景下可以超越CGLIB。
  • 安全性: 由于是Java原生支持,相对更安全。

3. 适用场景

JDK动态代理适用于目标对象实现了接口的场景。例如,Spring AOP在默认情况下会优先使用JDK动态代理来为实现了接口的Bean创建代理。

4. JDK动态代理示例

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;// 定义一个接口
interface UserService {void sayHello(String name);
}// 实现接口的类
class UserServiceImpl implements UserService {@Overridepublic void sayHello(String name) {System.out.println("Hello, " + name + " from UserServiceImpl");}
}// 代理处理器
class MyInvocationHandler implements InvocationHandler {private Object target;public MyInvocationHandler(Object target) {this.target = target;}@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {System.out.println("Before method: " + method.getName());Object result = method.invoke(target, args);System.out.println("After method: " + method.getName());return result;}
}public class JdkProxyDemo {public static void main(String[] args) {UserService userService = new UserServiceImpl();UserService proxy = (UserService) Proxy.newProxyInstance(userService.getClass().getClassLoader(),userService.getClass().getInterfaces(),new MyInvocationHandler(userService));proxy.sayHello("World");}
}

二、CGLIB动态代理

1. 原理

CGLIB(Code Generation Library)是一个强大的、高性能的字节码生成库。它可以在运行时扩展Java类和实现Java接口。CGLIB动态代理的原理是通过继承目标类来创建代理类。它使用ASM(一个Java字节码操作框架)来修改字节码,生成目标类的子类,并在子类中重写父类的方法,从而实现方法的增强。当代理对象的方法被调用时,会通过CGLIB生成的代理类来调用,并执行我们定义的增强逻辑。

2. 特点

  • 基于类: CGLIB动态代理可以代理没有实现接口的类。它通过继承目标类来实现代理,因此目标类不能是final类,因为final类无法被继承。
  • 运行时生成: 代理类在运行时动态生成,并加载到JVM中。
  • 性能: 在Java 8及以前的版本中,CGLIB通常比JDK动态代理具有更好的性能,因为它直接操作字节码,避免了反射的开销。但在Java 9及以后的版本中,JDK动态代理的性能已有所提升,两者的性能差距逐渐缩小。
  • 侵入性: 相对于JDK动态代理,CGLIB对目标类有一定的侵入性,因为它需要继承目标类。

3. 适用场景

CGLIB动态代理适用于目标对象没有实现接口的场景,或者需要代理final方法以外的任何方法。Spring AOP在目标Bean没有实现接口时,会使用CGLIB动态代理。

4. CGLIB动态代理示例

首先,确保你的项目中引入了CGLIB的依赖:

<dependency><groupId>cglib</groupId><artifactId>cglib</artifactId><version>3.3.0</version>
</dependency>
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
import java.lang.reflect.Method;// 没有实现接口的普通类
class ProductService {public void getProduct(String id) {System.out.println("Getting product with ID: " + id);}
}// 代理拦截器
class MyMethodInterceptor implements MethodInterceptor {@Overridepublic Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {System.out.println("Before method: " + method.getName());Object result = proxy.invokeSuper(obj, args);System.out.println("After method: " + method.getName());return result;}
}public class CglibProxyDemo {public static void main(String[] args) {Enhancer enhancer = new Enhancer();enhancer.setSuperclass(ProductService.class);enhancer.setCallback(new MyMethodInterceptor());ProductService proxy = (ProductService) enhancer.create();proxy.getProduct("123");}
}

三、JDK与CGLIB对比

特性JDK动态代理CGLIB动态代理
代理方式基于接口基于类继承
限制只能代理实现了接口的类不能代理final类和final方法
性能Java 8及以前版本通常低于CGLIB,Java 9+有所提升Java 8及以前版本通常优于JDK,Java 9+性能差距缩小
侵入性无侵入性对目标类有侵入性(继承)
原生支持Java原生支持第三方库支持

四、动态代理选择建议

在实际项目中,选择JDK动态代理还是CGLIB动态代理,主要取决于以下几个因素:

  1. 目标对象是否实现接口:

    • 如果目标对象实现了接口,并且你希望通过接口进行代理,那么JDK动态代理是首选。它简单、直接,并且是Java原生支持的。
    • 如果目标对象没有实现接口,或者你希望代理的是一个普通的类,那么CGLIB动态代理是唯一的选择。
  2. 性能要求:

    • 在大多数业务场景下,JDK和CGLIB的性能差异可以忽略不计。如果对性能有极致要求,并且在Java 8及以前的版本中,可以考虑CGLIB。但在Java 9及以后的版本中,建议进行实际测试,因为JDK动态代理的性能已大幅提升。
  3. 框架选择:

    • 许多主流框架(如Spring)会根据情况自动选择合适的代理方式。例如,Spring AOP在默认情况下,如果目标对象实现了接口,则使用JDK动态代理;否则,使用CGLIB动态代理。你也可以通过配置来强制指定代理方式。
  4. final类和final方法:

    • 如果目标类是final的,或者需要代理的方法是final的,那么CGLIB动态代理将无法工作,因为CGLIB通过继承来实现代理,而final类和final方法不能被继承或重写。在这种情况下,你可能需要重新考虑设计,或者寻找其他代理方案。

五、结论

JDK动态代理和CGLIB动态代理各有优劣,它们在Java生态系统中扮演着重要的角色。JDK动态代理是Java语言的内置功能,适用于基于接口的代理;而CGLIB则是一个强大的第三方库,适用于基于类的代理。在选择时,应根据目标对象的特性、性能要求以及所使用的框架等因素进行综合考量。

http://www.xdnf.cn/news/14225.html

相关文章:

  • 2.6 激光雷达消息格式
  • ESP32开发-ESP32P4环境配置
  • 【AD笔记】嘉立创元件导入到AD中(原理图-pcd-3D模型)
  • std::ifstream file(filename);详细解释
  • 十字滑台是否可以进行自动化控制?
  • window11等禁止系统更新的设置
  • 【数梦工场】【智慧航空AI大赛】比赛分享 阅读笔记
  • Hugging face 和 魔搭
  • 【论文阅读】Qwen2.5-VL Technical Report
  • Unity 对象层级处理小结
  • UI前端与大数据:如何构建实时数据分析系统?
  • 13_算法链与管道
  • 用于生成式新颖视图合成的密集 3D 场景完成
  • Hashcat使用教程:快速上手密码恢复工具
  • AUTOSAR图解==>AUTOSAR_SRS_OCUDriver
  • 力扣面试150题--添加与搜索单词 - 数据结构设计
  • Java延时
  • python中的模块化编程:日期模块、math算术模块、random模块
  • 温度对IO通信的影响
  • pythonday46
  • Python 标准库之 math 模块
  • 智慧水利可视化:水利水电工程数智化
  • 快速排序C++实现
  • IO扩展的一种简易方法
  • ECharts 图表生成示例
  • CentOS7报错:Cannot find a valid baseurl for repo: base/7/x86_64
  • day034-rsync异地容灾
  • org.springframework.cloud.openfeign 组件解释
  • JAVA实战开源项目:在线课程管理系统 (Vue+SpringBoot) 附源码
  • 超强人工智能解决方案套件InfiniSynapse:精准的业务理解、对各种数据源进行全模态联合智能分析--部署安装@Ubuntu22.04 @Docker