参考
https://blog.csdn.net/bigtree_3721/article/details/50774044
http://www.cnblogs.com/hustyangli/archive/2008/09/01/1281319.html
CGlib、JDK AOP介绍
Spring Framework reference 2.0.5 参考手册中文版 6.6. 代理机制 中是这样写的:Spring AOP部分使用JDK动态代理
或者CGLIB
来为目标对象创建代理。(建议尽量使用JDK的动态代理)
- 如果
被代理的目标对象实现了至少一个接口
,则会使用JDK动态代理
。所有该目标类型实现的接口都将被代理。JDK动态代理只能对实现了接口的类生成代理
,而不能针对类。原理是通过动态代理,生成接口的实例类和对象,从而进行对目标实例的代理。 若该目标对象没有实现任何接口,则创建一个CGLIB代理
。其实现原理类似于JDK动态代理,只是它在运行期间生成的代理对象是针对目标类扩展的子类
,覆盖其中的方法,因为是继承,所以目标类或方法最好不要声明成final
。CGLIB是高效的代码生成包,底层是依靠ASM(开源的java字节码编辑类库)操作字节码实现的,性能比JDK强
。
如果你希望强制使用CGLIB代理
,(例如:希望代理目标对象的所有方法,而不只是实现自接口的方法
)那也可以。但是需要考虑以下问题:
- 无法通知(advice)
Final 方法
,因为他们不能被覆写。 - 需要将CGLIB 2二进制发行包放在classpath下面,与之相较JDK本身就提供了动态代理,添加CGLIB库,SPRING_HOME/cglib/*.jar
- 在spring配置文件中加入
< aop:aspectj-autoproxy proxy-target-class="true"/>
这样使用CGLIB代理也就不会出现ClassCastException问题了,也可以在性能上有所提高,关键是对于代理对象是否继承接口可以统一使用
。
Spring是依靠什么来判断采用哪种代理策略来生成AOP代理呢?以下代码就是Spring的判断逻辑
//org.springframework.aop.framework.DefaultAopProxyFactory
//参数AdvisedSupport 是Spring AOP配置相关类
public AopProxy createAopProxy(AdvisedSupport advisedSupport)
throws AopConfigException {
//在此判断使用JDK动态代理还是CGLIB代理
if (advisedSupport.isOptimize() || advisedSupport.isProxyTargetClass()
|| hasNoUserSuppliedProxyInterfaces(advisedSupport)) {
if (!cglibAvailable) {
throw new AopConfigException(
"Cannot proxy target class because CGLIB2 is not available. "
+ "Add CGLIB to the class path or specify proxy interfaces.");
}
return CglibProxyFactory.createCglibProxy(advisedSupport);
} else {
return new JdkDynamicAopProxy(advisedSupport);
}
}
dvisedSupport.isOptimize()与advisedSupport.isProxyTargetClass()默认返回都是false,所以在默认情况下目标对象有没有实现接口
决定着Spring采取的策略,当然可以设置advisedSupport.isOptimize()或者advisedSupport.isProxyTargetClass()返回为true,这样无论目标对象有没有实现接口,Spring都会选择使用CGLIB代理。