使用到的字段fastClassInfo在init()
方法内部已经赋过值,
fci.f2其实是自动生成的代理类的索引类Coder$$EnhancerByCGLIB$$8e91f654$$FastClassByCGLIB$$4e5eb5aa
,
fci.i2值是1,
所以fci.f2.invoke(fci.i2, obj, args);
实际执行的是:
这里的var10000其实是自动生成的代理类Coder$$EnhancerByCGLIB$$8e91f654
的实例,所以接着调用的是
代理类Coder$$EnhancerByCGLIB$$8e91f654
的CGLIB$work$0()
方法:
这里的super指的是Coder类,所以super.work();
实际执行的是Coder类的work()方法:
综上所述,coder.work();
的调用顺序依次是:
代理类--->自定义方法拦截器--->代理类索引类getIndex()方法-->代理类索引类invoke()方法--->代理类--->被代理类。
4. JDK动态代理与CGLIB动态代理区别(面试常问)
关于JDK动态代理,可以查看上一篇博客:【深度思考】聊聊JDK动态代理原理。
了解了JDK动态代理和CGLIB动态代理的原理后,现在来比较下两者的区别,这也是面试时几乎必问的一道面试题。
-
使用JDK动态代理,被代理类必须要实现接口,使用CGLIB动态代理,被代理类可以不实现接口
原因分析:
JDK动态代理生成的代理类继承了java.lang.reflect.Proxy
,因为Java是单继承的,如果不通过实现接口的形式,
无法对类进行扩展。
CGLIB动态代理生成的代理类实际上是被代理类的子类,所以被代理类可以不实现接口。
-
自动生成类的数量不同
JDK动态代理只会生成1个代理类,一般情况下名称为:com.sun.proxy.$Proxy0
。
CGLIB动态代理会生成好几个类,核心的3个分别是:
1)代理类:被代理类的子类,名称格式为Coder$$EnhancerByCGLIB$$8e91f654
,包名和被代理类包名一致。
2)代理类的索引类:名称格式为Coder$$EnhancerByCGLIB$$8e91f654$$FastClassByCGLIB$$4e5eb5aa
,
包名和被代理类包名一致。
3)被代理类的索引类:名称格式为Coder$$FastClassByCGLIB$$398819d0
,包名和被代理类包名一致。
-
生成代理类技术不同
JDK动态代理使用JDK自带的ProxyGenerator类生成字节码文件。
CGLIB动态代理使用ASM框架生成字节码文件。
-
调用方式不同
JDK动态代理:代理类--->InvocationHandler.invoke()--->被代理类方法(用到了反射)。
CGLIB动态代理:代理类--->MethodInterceptor.intercept()--->代理类索引类getIndex()--->
代理类索引类invoke()--->代理类--->被代理类。(直接调用)
文章持续更新,欢迎关注微信公众号「申城异乡人」第一时间阅读!