都说 Cglib 创建的动态代理的运行性能比 JDK 动态代理能高出大概 10 倍,今日抱着怀疑精神验证了一下,发现情况有所不同,遂贴出实验结果,以供参考和讨论。

代码很简单,首先,定义一个 Test 接口,和一个实现 TestImpl 。Test 接口仅定义一个方法 test,对传入的 int 参数加 1 后返回。代码如下:

  1. package my.test;
  2.  
  3. public interface Test {
  4.  
  5. public int test(int i);
  6.  
  7. }
  1. package my.test;
  2.  
  3. public class TestImpl implements Test{
  4. public int test(int i) {
  5. return i+1;
  6. }
  7. }

然后,定义了三种代理的实现:装饰者模式实现的代理(decorator),JDK 动态代理(dynamic proxy) 和 Cglib 动态代理 (cglib proxy)。代码如下:

  1. package my.test;
  2.  
  3. public class DecoratorTest implements Test{
  4. private Test target;
  5.  
  6. public DecoratorTest(Test target) {
  7. this.target = target;
  8. }
  9.  
  10. public int test(int i) {
  11. return target.test(i);
  12. }
  13. }
  1. package my.test;
  2.  
  3. import java.lang.reflect.InvocationHandler;
  4. import java.lang.reflect.Method;
  5. import java.lang.reflect.Proxy;
  6.  
  7. public class DynamicProxyTest implements InvocationHandler {
  8. private Test target;
  9.  
  10. private DynamicProxyTest(Test target) {
  11. this.target = target;
  12. }
  13.  
  14. public static Test newProxyInstance(Test target) {
  15. return (Test) Proxy
  16. .newProxyInstance(DynamicProxyTest.class.getClassLoader(),
  17. new Class<?>[] { Test.class },
  18. new DynamicProxyTest(target));
  19.  
  20. }
  21.  
  22. public Object invoke(Object proxy, Method method, Object[] args)
  23. throws Throwable {
  24. return method.invoke(target, args);
  25. }
  26. }
  1. package my.test;
  2.  
  3. import java.lang.reflect.Method;
  4.  
  5. import net.sf.cglib.proxy.Enhancer;
  6. import net.sf.cglib.proxy.MethodInterceptor;
  7. import net.sf.cglib.proxy.MethodProxy;
  8.  
  9. public class CglibProxyTest implements MethodInterceptor {
  10.  
  11. private CglibProxyTest() {
  12. }
  13.  
  14. public static <T extends Test> Test newProxyInstance(Class<T> targetInstanceClazz){
  15. Enhancer enhancer = new Enhancer();
  16. enhancer.setSuperclass(targetInstanceClazz);
  17. enhancer.setCallback(new CglibProxyTest());
  18. return (Test) enhancer.create();
  19. }
  20.  
  21. public Object intercept(Object obj, Method method, Object[] args,
  22. MethodProxy proxy) throws Throwable {
  23. return proxy.invokeSuper(obj, args);
  24. }
  25.  
  26. }

以 TestImpl 的调用耗时作为基准,对比通过其它三种代理进行调用的耗时。测试代码如下:

  1. package my.test;
  2.  
  3. import java.util.LinkedHashMap;
  4. import java.util.Map;
  5.  
  6. public class ProxyPerfTester {
  7.  
  8. public static void main(String[] args) {
  9. //创建测试对象;
  10. Test nativeTest = new TestImpl();
  11. Test decorator = new DecoratorTest(nativeTest);
  12. Test dynamicProxy = DynamicProxyTest.newProxyInstance(nativeTest);
  13. Test cglibProxy = CglibProxyTest.newProxyInstance(TestImpl.class);
  14.  
  15. //预热一下;
  16. int preRunCount = 10000;
  17. runWithoutMonitor(nativeTest, preRunCount);
  18. runWithoutMonitor(decorator, preRunCount);
  19. runWithoutMonitor(cglibProxy, preRunCount);
  20. runWithoutMonitor(dynamicProxy, preRunCount);
  21.  
  22. //执行测试;
  23. Map<String, Test> tests = new LinkedHashMap<String, Test>();
  24. tests.put("Native ", nativeTest);
  25. tests.put("Decorator", decorator);
  26. tests.put("Dynamic ", dynamicProxy);
  27. tests.put("Cglib ", cglibProxy);
  28. int repeatCount = 3;
  29. int runCount = 1000000;
  30. runTest(repeatCount, runCount, tests);
  31. runCount = 50000000;
  32. runTest(repeatCount, runCount, tests);
  33. }
  34.  
  35. private static void runTest(int repeatCount, int runCount, Map<String, Test> tests){
  36. System.out.println(String.format("\n==================== run test : [repeatCount=%s] [runCount=%s] [java.version=%s] ====================", repeatCount, runCount, System.getProperty("java.version")));
  37. for (int i = 0; i < repeatCount; i++) {
  38. System.out.println(String.format("\n--------- test : [%s] ---------", (i+1)));
  39. for (String key : tests.keySet()) {
  40. runWithMonitor(tests.get(key), runCount, key);
  41. }
  42. }
  43. }
  44.  
  45. private static void runWithoutMonitor(Test test, int runCount) {
  46. for (int i = 0; i < runCount; i++) {
  47. test.test(i);
  48. }
  49. }
  50.  
  51. private static void runWithMonitor(Test test, int runCount, String tag) {
  52. long start = System.currentTimeMillis();
  53. for (int i = 0; i < runCount; i++) {
  54. test.test(i);
  55. }
  56. long end = System.currentTimeMillis();
  57. System.out.println("["+tag + "] Elapsed Time:" + (end-start) + "ms");
  58. }
  59. }

测试用例分别在 jdk6、 jdk7、jdk8 下进行了测试,每次测试分别以 1,000,000 和 50,000,000 循环次数调用 test 方法,并重复3次。

  • jdk6 下的测试结果如下:
  1. ==================== run test : [repeatCount=3] [runCount=1000000] [java.version=1.6.0_45] ====================
  2.  
  3. --------- test : [1] ---------
  4. [Native ] Elapsed Time:2ms
  5. [Decorator] Elapsed Time:12ms
  6. [Dynamic ] Elapsed Time:31ms
  7. [Cglib ] Elapsed Time:31ms
  8.  
  9. --------- test : [2] ---------
  10. [Native ] Elapsed Time:7ms
  11. [Decorator] Elapsed Time:7ms
  12. [Dynamic ] Elapsed Time:31ms
  13. [Cglib ] Elapsed Time:27ms
  14.  
  15. --------- test : [3] ---------
  16. [Native ] Elapsed Time:7ms
  17. [Decorator] Elapsed Time:6ms
  18. [Dynamic ] Elapsed Time:23ms
  19. [Cglib ] Elapsed Time:29ms
  20.  
  21. ==================== run test : [repeatCount=3] [runCount=50000000] [java.version=1.6.0_45] ====================
  22.  
  23. --------- test : [1] ---------
  24. [Native ] Elapsed Time:212ms
  25. [Decorator] Elapsed Time:226ms
  26. [Dynamic ] Elapsed Time:1054ms
  27. [Cglib ] Elapsed Time:830ms
  28.  
  29. --------- test : [2] ---------
  30. [Native ] Elapsed Time:184ms
  31. [Decorator] Elapsed Time:222ms
  32. [Dynamic ] Elapsed Time:1020ms
  33. [Cglib ] Elapsed Time:826ms
  34.  
  35. --------- test : [3] ---------
  36. [Native ] Elapsed Time:184ms
  37. [Decorator] Elapsed Time:208ms
  38. [Dynamic ] Elapsed Time:979ms
  39. [Cglib ] Elapsed Time:832ms

  测试结果表明:jdk6 下,在运行次数较少的情况下,jdk动态代理与 cglib 差距不明显,甚至更快一些;而当调用次数增加之后, cglib 表现稍微更快一些,然而仅仅是“稍微”好一些,远没达到 10 倍差距。

  • jdk7 下的测试结果如下:
  1. ==================== run test : [repeatCount=3] [runCount=1000000] [java.version=1.7.0_60] ====================
  2.  
  3. --------- test : [1] ---------
  4. [Native ] Elapsed Time:2ms
  5. [Decorator] Elapsed Time:12ms
  6. [Dynamic ] Elapsed Time:19ms
  7. [Cglib ] Elapsed Time:26ms
  8.  
  9. --------- test : [2] ---------
  10. [Native ] Elapsed Time:3ms
  11. [Decorator] Elapsed Time:5ms
  12. [Dynamic ] Elapsed Time:17ms
  13. [Cglib ] Elapsed Time:20ms
  14.  
  15. --------- test : [3] ---------
  16. [Native ] Elapsed Time:4ms
  17. [Decorator] Elapsed Time:4ms
  18. [Dynamic ] Elapsed Time:13ms
  19. [Cglib ] Elapsed Time:27ms
  20.  
  21. ==================== run test : [repeatCount=3] [runCount=50000000] [java.version=1.7.0_60] ====================
  22.  
  23. --------- test : [1] ---------
  24. [Native ] Elapsed Time:208ms
  25. [Decorator] Elapsed Time:210ms
  26. [Dynamic ] Elapsed Time:551ms
  27. [Cglib ] Elapsed Time:923ms
  28.  
  29. --------- test : [2] ---------
  30. [Native ] Elapsed Time:238ms
  31. [Decorator] Elapsed Time:210ms
  32. [Dynamic ] Elapsed Time:483ms
  33. [Cglib ] Elapsed Time:872ms
  34.  
  35. --------- test : [3] ---------
  36. [Native ] Elapsed Time:217ms
  37. [Decorator] Elapsed Time:208ms
  38. [Dynamic ] Elapsed Time:494ms
  39. [Cglib ] Elapsed Time:881ms

测试结果表明:jdk7 下,情况发生了逆转!在运行次数较少(1,000,000)的情况下,jdk动态代理比 cglib 快了差不多30%;而当调用次数增加之后(50,000,000), 动态代理比 cglib 快了接近1倍。

接下来再看看jdk8下的表现如何。

  • jdk8 下的测试结果如下:
  1. ==================== run test : [repeatCount=3] [runCount=1000000] [java.version=1.8.0_05] ====================
  2.  
  3. --------- test : [1] ---------
  4. [Native ] Elapsed Time:5ms
  5. [Decorator] Elapsed Time:11ms
  6. [Dynamic ] Elapsed Time:27ms
  7. [Cglib ] Elapsed Time:52ms
  8.  
  9. --------- test : [2] ---------
  10. [Native ] Elapsed Time:4ms
  11. [Decorator] Elapsed Time:6ms
  12. [Dynamic ] Elapsed Time:11ms
  13. [Cglib ] Elapsed Time:24ms
  14.  
  15. --------- test : [3] ---------
  16. [Native ] Elapsed Time:4ms
  17. [Decorator] Elapsed Time:5ms
  18. [Dynamic ] Elapsed Time:9ms
  19. [Cglib ] Elapsed Time:26ms
  20.  
  21. ==================== run test : [repeatCount=3] [runCount=50000000] [java.version=1.8.0_05] ====================
  22.  
  23. --------- test : [1] ---------
  24. [Native ] Elapsed Time:194ms
  25. [Decorator] Elapsed Time:211ms
  26. [Dynamic ] Elapsed Time:538ms
  27. [Cglib ] Elapsed Time:965ms
  28.  
  29. --------- test : [2] ---------
  30. [Native ] Elapsed Time:194ms
  31. [Decorator] Elapsed Time:214ms
  32. [Dynamic ] Elapsed Time:503ms
  33. [Cglib ] Elapsed Time:969ms
  34.  
  35. --------- test : [3] ---------
  36. [Native ] Elapsed Time:190ms
  37. [Decorator] Elapsed Time:209ms
  38. [Dynamic ] Elapsed Time:495ms
  39. [Cglib ] Elapsed Time:939ms

测试结果表明:jdk8 下,延续了 JDK7 下的惊天大逆转!不过还观察另外有一个细微的变化,从绝对值来看 cglib 在 jdk8 下的表现似乎比 jdk7 还要差一点点,尽管只是一点点,但经过反复多次的执行仍然是这个趋势(注:这个趋势的结论并不严谨,只是捎带一提,如需得出结论还需进行更多样的对比实验)。

结论:从 jdk6 到 jdk7、jdk8 ,动态代理的性能得到了显著的提升,而 cglib 的表现并未跟上,甚至可能会略微下降。传言的 cglib 比 jdk动态代理高出 10 倍的情况也许是出现在更低版本的 jdk 上吧。

以上测试用例虽然简单,但揭示了 jdk 版本升级可能会带来一些新技术改变,会使我们以前的经验失效。放在真实业务场景下时,还需要按照实际情况进行测试后才能得出特定于场景的结论。

总之,实践出真知,还要与时俱进地去检视更新一些以往经验。

注:上述实验中 cglib 的版本是 3.1 。

Cglib 与 JDK动态代理的运行性能比较的更多相关文章

  1. Cglib 与 JDK动态代理

    作者:xiaolyuh 时间:2019/09/20 09:58 AOP 代理的两种实现: jdk是代理接口,私有方法必然不会存在在接口里,所以就不会被拦截到: cglib是子类,private的方法照 ...

  2. 学习CGLIB与JDK动态代理的区别

    动态代理 代理模式是Java中常见的一种模式.代理又分为静态代理和动态代理.静态代理就是显式指定的代理,静态代理的优点是由程序员自行指定代理类并进行编译和运行,缺点是一个代理类只能对一个接口的实现类进 ...

  3. 面试造火箭系列,栽在了cglib和jdk动态代理

    "喂,你好,我是XX巴巴公司的技术面试官,请问你是张小帅吗".声音是从电话那头传来的 "是的,你好".小帅暗喜,大厂终于找上我了. "下面我们来进行一 ...

  4. CGlib和JDK动态代理

    一.CGlib动态代理     JDK实现动态代理需要实现类通过接口定义业务方法,对于没有接口的类,如何实现动态代理呢,这就需要CGLib了.CGLib采用了非常底层的1:字节码技术,其原理是通过字节 ...

  5. 输出cglib以及jdk动态代理产生的class文件

      --该设置用于输出jdk动态代理产生的类 System.getProperties().put("sun.misc.ProxyGenerator.saveGeneratedFiles&q ...

  6. spring cglib 与 jdk 动态代理

    1. 概述 JDK动态代理是利用java反射机制 生成一个实现接口的匿名类, 在调用具体方法前调用InvocationHandler来处理 Cglib动态代理是 利用asm开源包 把被代理类的clas ...

  7. 有点深度的聊聊JDK动态代理

    在接触SpringAOP的时候,大家一定会被这神奇的功能所折服,想知道其中的奥秘,底层到底是如何实现的.于是,大家会通过搜索引擎,知道了一个陌生的名词:动态代理,慢慢的又知道了动态代理有多种实现方式, ...

  8. [编织消息框架][JAVA核心技术]jdk动态代理

    需要用到的工具  jdk : javac javap class 反编译 :JD-GUI http://jd.benow.ca/ import java.lang.reflect.Invocation ...

  9. java学习笔记(中级篇)—JDK动态代理

    一.什么是代理模式 相信大家都知道代理商这个概念,在商业中,代理商无处不在.假设你要去买东西,你不可能去找真正的厂家去买,也不可能直接跟厂家提出需求,代理商就是这中间的一桥梁,连接买家和厂商.你要买或 ...

随机推荐

  1. [原]unity3d GLSL无法pc上显示

    pc编写GLSL: Shader "Custom/GLSL" {    Properties {        _MainTex ("Base (RGB)", ...

  2. 关于解决emoji表情的存储

    近段时间处理,由于工作需求,需要使得用户插入的emoji表情能够正常显示及使用,所以做个总结,以备后用. 说明:本方法只在mysql环境中测试 1.首先程序在连接数据库时,要指定数据库字符集的设置 c ...

  3. 3D物体识别的如果检验

    3D物体识别的如果验证 这次目的在于解释怎样做3D物体识别通过验证模型如果在聚类里面.在描写叙述器匹配后,这次我们将执行某个相关组算法在PCL里面为了聚类点对点相关性的集合,决定如果物体在场景里面的实 ...

  4. Java -- 异常的捕获及处理 -- 范例 -- throw与throws的应用

    7.2.3 范例 -- throw与throws的应用 例:综合应用 Class : Math package limeThrowable._7_2_3; public class Math { pu ...

  5. mongodb 初学 目录

    mongodb 初学 索引 啦啦啦 MongoDB 教程 NoSQL 简介 MongoDB 简介 Windows 平台安装 MongoDB Linux平台安装MongoDB mongodb 在 Ubu ...

  6. ios开发之--新手引导页图片适配方案

    1,图片适配,最早以前是自己命名规范,例如@1x,@2x,@3x等,3套图基本上就够用了 2,在iPhone X之后,需要适配的图就多了,因为分辨率增多了,屏幕尺寸也增多了 3,尺寸 :640*960 ...

  7. Docker应用之仓库

    仓库是存放镜像的地方 注册服务器是管理仓库的具体服务器,每个服务器上可以有多个仓库,每个仓库也可以有多个镜像 如 dl.dockerpool.com/ubuntu ,dl.dockerpool.com ...

  8. IOPS性能指标

    如何计算mysql的IOPS? qps 每秒处理的查询数tps 每秒处理的事务数IOPS,每秒磁盘进行的I/O操作次数 今天看到一篇文章说磁盘理论最大IOPS为200左右,我有两个疑问:1.MYSQL ...

  9. NFS 简介

    一.NFS 简介 1. NFS(Network File System)网络文件系统,它允许网络中的计算机之间通过 TCP/IP 网络共享资源2. NFS 应用场景:A,B,C 三台机器上需要保证被访 ...

  10. Linux下getsockopt/setsockopt 函数说明

    [ getsockopt/setsockopt系统调用 功能描述:  获取或者设置与某个套接字关联的选 项.选项可能存在于多层协议中,它们总会出现在最上面的套接字层.当操作套接字选项时,选项位于的层和 ...