Cglib 与 JDK动态代理的运行性能比较
都说 Cglib 创建的动态代理的运行性能比 JDK 动态代理能高出大概 10 倍,今日抱着怀疑精神验证了一下,发现情况有所不同,遂贴出实验结果,以供参考和讨论。
代码很简单,首先,定义一个 Test 接口,和一个实现 TestImpl 。Test 接口仅定义一个方法 test,对传入的 int 参数加 1 后返回。代码如下:
package my.test;
public interface Test {
public int test(int i);
}
package my.test;
public class TestImpl implements Test{
public int test(int i) {
return i+1;
}
}
然后,定义了三种代理的实现:装饰者模式实现的代理(decorator),JDK 动态代理(dynamic proxy) 和 Cglib 动态代理 (cglib proxy)。代码如下:
package my.test;
public class DecoratorTest implements Test{
private Test target;
public DecoratorTest(Test target) {
this.target = target;
}
public int test(int i) {
return target.test(i);
}
}
package my.test; import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy; public class DynamicProxyTest implements InvocationHandler {
private Test target; private DynamicProxyTest(Test target) {
this.target = target;
} public static Test newProxyInstance(Test target) {
return (Test) Proxy
.newProxyInstance(DynamicProxyTest.class.getClassLoader(),
new Class<?>[] { Test.class },
new DynamicProxyTest(target)); } public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
return method.invoke(target, args);
}
}
package my.test; import java.lang.reflect.Method; import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy; public class CglibProxyTest implements MethodInterceptor { private CglibProxyTest() {
} public static <T extends Test> Test newProxyInstance(Class<T> targetInstanceClazz){
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(targetInstanceClazz);
enhancer.setCallback(new CglibProxyTest());
return (Test) enhancer.create();
} public Object intercept(Object obj, Method method, Object[] args,
MethodProxy proxy) throws Throwable {
return proxy.invokeSuper(obj, args);
} }
以 TestImpl 的调用耗时作为基准,对比通过其它三种代理进行调用的耗时。测试代码如下:
package my.test; import java.util.LinkedHashMap;
import java.util.Map; public class ProxyPerfTester { public static void main(String[] args) {
//创建测试对象;
Test nativeTest = new TestImpl();
Test decorator = new DecoratorTest(nativeTest);
Test dynamicProxy = DynamicProxyTest.newProxyInstance(nativeTest);
Test cglibProxy = CglibProxyTest.newProxyInstance(TestImpl.class); //预热一下;
int preRunCount = 10000;
runWithoutMonitor(nativeTest, preRunCount);
runWithoutMonitor(decorator, preRunCount);
runWithoutMonitor(cglibProxy, preRunCount);
runWithoutMonitor(dynamicProxy, preRunCount); //执行测试;
Map<String, Test> tests = new LinkedHashMap<String, Test>();
tests.put("Native ", nativeTest);
tests.put("Decorator", decorator);
tests.put("Dynamic ", dynamicProxy);
tests.put("Cglib ", cglibProxy);
int repeatCount = 3;
int runCount = 1000000;
runTest(repeatCount, runCount, tests);
runCount = 50000000;
runTest(repeatCount, runCount, tests);
} private static void runTest(int repeatCount, int runCount, Map<String, Test> tests){
System.out.println(String.format("\n==================== run test : [repeatCount=%s] [runCount=%s] [java.version=%s] ====================", repeatCount, runCount, System.getProperty("java.version")));
for (int i = 0; i < repeatCount; i++) {
System.out.println(String.format("\n--------- test : [%s] ---------", (i+1)));
for (String key : tests.keySet()) {
runWithMonitor(tests.get(key), runCount, key);
}
}
} private static void runWithoutMonitor(Test test, int runCount) {
for (int i = 0; i < runCount; i++) {
test.test(i);
}
} private static void runWithMonitor(Test test, int runCount, String tag) {
long start = System.currentTimeMillis();
for (int i = 0; i < runCount; i++) {
test.test(i);
}
long end = System.currentTimeMillis();
System.out.println("["+tag + "] Elapsed Time:" + (end-start) + "ms");
}
}
测试用例分别在 jdk6、 jdk7、jdk8 下进行了测试,每次测试分别以 1,000,000 和 50,000,000 循环次数调用 test 方法,并重复3次。
- jdk6 下的测试结果如下:
==================== run test : [repeatCount=3] [runCount=1000000] [java.version=1.6.0_45] ==================== --------- test : [1] ---------
[Native ] Elapsed Time:2ms
[Decorator] Elapsed Time:12ms
[Dynamic ] Elapsed Time:31ms
[Cglib ] Elapsed Time:31ms --------- test : [2] ---------
[Native ] Elapsed Time:7ms
[Decorator] Elapsed Time:7ms
[Dynamic ] Elapsed Time:31ms
[Cglib ] Elapsed Time:27ms --------- test : [3] ---------
[Native ] Elapsed Time:7ms
[Decorator] Elapsed Time:6ms
[Dynamic ] Elapsed Time:23ms
[Cglib ] Elapsed Time:29ms ==================== run test : [repeatCount=3] [runCount=50000000] [java.version=1.6.0_45] ==================== --------- test : [1] ---------
[Native ] Elapsed Time:212ms
[Decorator] Elapsed Time:226ms
[Dynamic ] Elapsed Time:1054ms
[Cglib ] Elapsed Time:830ms --------- test : [2] ---------
[Native ] Elapsed Time:184ms
[Decorator] Elapsed Time:222ms
[Dynamic ] Elapsed Time:1020ms
[Cglib ] Elapsed Time:826ms --------- test : [3] ---------
[Native ] Elapsed Time:184ms
[Decorator] Elapsed Time:208ms
[Dynamic ] Elapsed Time:979ms
[Cglib ] Elapsed Time:832ms
测试结果表明:jdk6 下,在运行次数较少的情况下,jdk动态代理与 cglib 差距不明显,甚至更快一些;而当调用次数增加之后, cglib 表现稍微更快一些,然而仅仅是“稍微”好一些,远没达到 10 倍差距。
- jdk7 下的测试结果如下:
==================== run test : [repeatCount=3] [runCount=1000000] [java.version=1.7.0_60] ==================== --------- test : [1] ---------
[Native ] Elapsed Time:2ms
[Decorator] Elapsed Time:12ms
[Dynamic ] Elapsed Time:19ms
[Cglib ] Elapsed Time:26ms --------- test : [2] ---------
[Native ] Elapsed Time:3ms
[Decorator] Elapsed Time:5ms
[Dynamic ] Elapsed Time:17ms
[Cglib ] Elapsed Time:20ms --------- test : [3] ---------
[Native ] Elapsed Time:4ms
[Decorator] Elapsed Time:4ms
[Dynamic ] Elapsed Time:13ms
[Cglib ] Elapsed Time:27ms ==================== run test : [repeatCount=3] [runCount=50000000] [java.version=1.7.0_60] ==================== --------- test : [1] ---------
[Native ] Elapsed Time:208ms
[Decorator] Elapsed Time:210ms
[Dynamic ] Elapsed Time:551ms
[Cglib ] Elapsed Time:923ms --------- test : [2] ---------
[Native ] Elapsed Time:238ms
[Decorator] Elapsed Time:210ms
[Dynamic ] Elapsed Time:483ms
[Cglib ] Elapsed Time:872ms --------- test : [3] ---------
[Native ] Elapsed Time:217ms
[Decorator] Elapsed Time:208ms
[Dynamic ] Elapsed Time:494ms
[Cglib ] Elapsed Time:881ms
测试结果表明:jdk7 下,情况发生了逆转!在运行次数较少(1,000,000)的情况下,jdk动态代理比 cglib 快了差不多30%;而当调用次数增加之后(50,000,000), 动态代理比 cglib 快了接近1倍。
接下来再看看jdk8下的表现如何。
- jdk8 下的测试结果如下:
==================== run test : [repeatCount=3] [runCount=1000000] [java.version=1.8.0_05] ==================== --------- test : [1] ---------
[Native ] Elapsed Time:5ms
[Decorator] Elapsed Time:11ms
[Dynamic ] Elapsed Time:27ms
[Cglib ] Elapsed Time:52ms --------- test : [2] ---------
[Native ] Elapsed Time:4ms
[Decorator] Elapsed Time:6ms
[Dynamic ] Elapsed Time:11ms
[Cglib ] Elapsed Time:24ms --------- test : [3] ---------
[Native ] Elapsed Time:4ms
[Decorator] Elapsed Time:5ms
[Dynamic ] Elapsed Time:9ms
[Cglib ] Elapsed Time:26ms ==================== run test : [repeatCount=3] [runCount=50000000] [java.version=1.8.0_05] ==================== --------- test : [1] ---------
[Native ] Elapsed Time:194ms
[Decorator] Elapsed Time:211ms
[Dynamic ] Elapsed Time:538ms
[Cglib ] Elapsed Time:965ms --------- test : [2] ---------
[Native ] Elapsed Time:194ms
[Decorator] Elapsed Time:214ms
[Dynamic ] Elapsed Time:503ms
[Cglib ] Elapsed Time:969ms --------- test : [3] ---------
[Native ] Elapsed Time:190ms
[Decorator] Elapsed Time:209ms
[Dynamic ] Elapsed Time:495ms
[Cglib ] Elapsed Time:939ms
测试结果表明:jdk8 下,延续了 JDK7 下的惊天大逆转!不过还观察另外有一个细微的变化,从绝对值来看 cglib 在 jdk8 下的表现似乎比 jdk7 还要差一点点,尽管只是一点点,但经过反复多次的执行仍然是这个趋势(注:这个趋势的结论并不严谨,只是捎带一提,如需得出结论还需进行更多样的对比实验)。
结论:从 jdk6 到 jdk7、jdk8 ,动态代理的性能得到了显著的提升,而 cglib 的表现并未跟上,甚至可能会略微下降。传言的 cglib 比 jdk动态代理高出 10 倍的情况也许是出现在更低版本的 jdk 上吧。
以上测试用例虽然简单,但揭示了 jdk 版本升级可能会带来一些新技术改变,会使我们以前的经验失效。放在真实业务场景下时,还需要按照实际情况进行测试后才能得出特定于场景的结论。
总之,实践出真知,还要与时俱进地去检视更新一些以往经验。
注:上述实验中 cglib 的版本是 3.1 。
Cglib 与 JDK动态代理的运行性能比较的更多相关文章
- Cglib 与 JDK动态代理
作者:xiaolyuh 时间:2019/09/20 09:58 AOP 代理的两种实现: jdk是代理接口,私有方法必然不会存在在接口里,所以就不会被拦截到: cglib是子类,private的方法照 ...
- 学习CGLIB与JDK动态代理的区别
动态代理 代理模式是Java中常见的一种模式.代理又分为静态代理和动态代理.静态代理就是显式指定的代理,静态代理的优点是由程序员自行指定代理类并进行编译和运行,缺点是一个代理类只能对一个接口的实现类进 ...
- 面试造火箭系列,栽在了cglib和jdk动态代理
"喂,你好,我是XX巴巴公司的技术面试官,请问你是张小帅吗".声音是从电话那头传来的 "是的,你好".小帅暗喜,大厂终于找上我了. "下面我们来进行一 ...
- CGlib和JDK动态代理
一.CGlib动态代理 JDK实现动态代理需要实现类通过接口定义业务方法,对于没有接口的类,如何实现动态代理呢,这就需要CGLib了.CGLib采用了非常底层的1:字节码技术,其原理是通过字节 ...
- 输出cglib以及jdk动态代理产生的class文件
--该设置用于输出jdk动态代理产生的类 System.getProperties().put("sun.misc.ProxyGenerator.saveGeneratedFiles&q ...
- spring cglib 与 jdk 动态代理
1. 概述 JDK动态代理是利用java反射机制 生成一个实现接口的匿名类, 在调用具体方法前调用InvocationHandler来处理 Cglib动态代理是 利用asm开源包 把被代理类的clas ...
- 有点深度的聊聊JDK动态代理
在接触SpringAOP的时候,大家一定会被这神奇的功能所折服,想知道其中的奥秘,底层到底是如何实现的.于是,大家会通过搜索引擎,知道了一个陌生的名词:动态代理,慢慢的又知道了动态代理有多种实现方式, ...
- [编织消息框架][JAVA核心技术]jdk动态代理
需要用到的工具 jdk : javac javap class 反编译 :JD-GUI http://jd.benow.ca/ import java.lang.reflect.Invocation ...
- java学习笔记(中级篇)—JDK动态代理
一.什么是代理模式 相信大家都知道代理商这个概念,在商业中,代理商无处不在.假设你要去买东西,你不可能去找真正的厂家去买,也不可能直接跟厂家提出需求,代理商就是这中间的一桥梁,连接买家和厂商.你要买或 ...
随机推荐
- Tomcat------如何查看80端口是否被占用
1.Window + R,打开“运行”窗口,输入cmd 2.输入netstat -nao,回车,一般来说80端口会被PID为4的程序占用 3.启动任务管理器,点击“查看”->“选择列” 4.勾选 ...
- Java实现文件批量重命名
Windows操作系统可以实现重命名文件操作,却不能实现批量重命名.本实例实现了批量重命名功能,可以将一个文件夹内同一类型的文件按照一定的规则批量重命名.用户可以给出重命名模板,程序可以根据模板对相应 ...
- webstorm批量查找,批量替换快捷键
ctrl+shift+f:批量查找,我的webstorm11不能用ctrl+shift+f进行批量查找了,不知道什么原因,自己又胡乱实验了一下, 发现ctrl+shift+g+f可以批量查找 ctrl ...
- python3存入redis是bytes
在python3 中使用redis存储数据,存进去的是bytes >>> import redis >>> import time >>> imp ...
- D盾 v2.0.6.42 测试记录
0x01 前言 之前发了一篇博客<Bypass D盾_IIS防火墙SQL注入防御(多姿势)>,D哥第一时间联系我,对问题进行修复.这段时间与D哥聊了挺多关于D盾这款产品的话题,实在是很佩服 ...
- iOS开发-修改UITableViewCell中image和title的位置和大小
最近在开发中遇到需要Cell中imageView和textLable位置和大小的情况,设计希望得到的结果如下图所示: 而TableViewCell默认样式,image是靠紧左边的,并且image和ti ...
- C 预处理
http://baike.baidu.com/link?url=0mwKZRcxHuNHa_TiwXgpQPS2S-YbOGCUJVSgZ9sb-qe-G-x4oIDZpWuZqiVNBsMYA4HT ...
- Android 自定义 View 浅析
Android 自定义 View 浅析 概括 说到自定义 View ,就一定得说说 android 系统的UI绘制流程.再说这个流程之前,我们先看一下在每一个 activity 页面中我们的布局 ui ...
- 批量更改数据库表架构(生成sql后直接执行!)
批量更改数据库表架构(生成sql后直接执行!) use my_test; --当前数据库 ), ), ), @NewSql VARCHAR(max), @Index INT; SET @SchemaO ...
- 【大数据系列】HDFS安全模式
一.什么是安全模式 安全模式时HDFS所处的一种特殊状态,在这种状态下,文件系统只接受读数据请求,而不接受删除.修改等变更请求.在NameNode主节点启动时,HDFS首先进入安全模式,DataNod ...