参考
https://blog.csdn.net/bigtree_3721/article/details/50774044
http://www.cnblogs.com/hustyangli/archive/2008/09/01/1281319.html
Spring AOP部分使用JDK动态代理
或者CGLIB
来为目标对象创建代理。(建议尽量使用JDK的动态代理)
如果被代理的目标对象实现了至少一个接口
,则会使用JDK动态代理
。所有该目标类型实现的接口都将被代理。若该目标对象没有实现任何接口,则创建一个CGLIB代理
。
如果你希望强制使用CGLIB代理
,(例如:希望代理目标对象的所有方法,而不只是实现自接口的方法
)那也可以。但是需要考虑以下问题:
Final 方法
,因为他们不能被覆写。强制使用CGLIB代理
需要将 < aop:config> 的 proxy-target-class 属性设为true
:
< aop:config proxy-target-class="true">
...
< /aop:config>
当需要使用CGLIB代理和@AspectJ自动代理支持
,请按照如下的方式设置 < aop:aspectj-autoproxy> 的 proxy-target-class 属性:
< aop:aspectj-autoproxy proxy-target-class="true"/>
而实际使用的过程中才会发现细节问题的差别,The devil is in the detail.
JDK动态代理
:其代理对象必须是某个接口的实现
,它是通过在运行期间创建一个接口的实现类
来完成对目标对象的代理。
CGLIB代理:实现原理类似于JDK动态代理,只是它在运行期间生成的代理对象是针对目标类扩展的子类
。CGLIB是高效的代码生成包,底层是依靠ASM(开源的java字节码编辑类库)操作字节码实现的,性能比JDK强
。
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代理。
所以在默认情况下,如果一个目标对象如果实现了接口,
Spring则会选择JDK动态代理
策略动态的创建一个接口实现类(动态代理类)
来代理目标对象,可以通俗的理解这个动态代理类是目标对象的另外一个版本
。
所以在默认情况下,如果目标对象没有实现任何接口,Spring会选择CGLIB代理
,其生成的动态代理对象是目标类的子类
。
如果当需要使用CGLIB代理和@AspectJ自动代理支持,请按照如下的方式设置 < aop:aspectj-autoproxy> 的 proxy-target-class 属性:
< aop:aspectj-autoproxy proxy-target-class="true"/>
这样使用CGLIB代理也就不会出现前面提到的ClassCastException问题了,也可以在性能上有所提高,关键是对于代理对象是否继承接口可以统一使用
。
Aspect默认情况下不用实现接口
,但对于目标对象
(UserManagerImpl.java),在默认情况下必须实现接口
,如果没有实现接口必须引入CGLIB库,我们可以通过Advice中添加一个JoinPoint参数,这个值会由spring自动传入,从JoinPoint中可以取得参数值、方法名等等
目标对象没有实现接口
,必须采用CGLIB库
,spring会自动在JDK动态代理和CGLIB之间转换如何强制使用CGLIB实现AOP?
JDK动态代理和CGLIB字节码生成的区别?
只能对实现了接口的类生成代理
,而不能针对类。原理是通过动态代理,生成接口的实例类和对象,从而进行对目标实例的代理。针对类实现代理
,主要是对指定的类生成一个子类,覆盖其中的方法
因为是继承,所以目标类或方法最好不要声明成final
。