spring 如何决定使用jdk动态代理和cglib
Spring AOP部分使用JDK动态代理或者CGLIB来为目标对象创建代理。(建议尽量使用JDK的动态代理)
如果被代理的目标对象实现了至少一个接口,则会使用JDK动态代理。所有该目标类型实现的接口都将被代理。若该目标对象没有实现任何接口,则创建一个CGLIB代理。
如果你希望强制使用CGLIB代理,(例如:希望代理目标对象的所有方法,而不只是实现自接口的方法)那也可以。但是需要考虑以下问题:
无法通知(advise)Final 方法,因为他们不能被覆写。
你需要将CGLIB 2二进制发行包放在classpath下面,与之相较JDK本身就提供了动态代理
强制使用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);
}
}
advisedSupport.isOptimize()与advisedSupport.isProxyTargetClass()默认返回都是
false,所以在默认情况下目标对象有没有实现接口决定着Spring采取的策略,当然可以设置
advisedSupport.isOptimize()或者advisedSupport.isProxyTargetClass()返回为true,
这样无论目标对象有没有实现接口Spring都会选择使用CGLIB代理。
所以在默认情况下,如果一个目标对象如果实现了接口
Spring则会选择JDK动态代理策略动态的创建一个接口实现类(动态代理类)来代理目标对象,可以通俗的理解这个动态代理类是目标对象的另外一个版
本,所以这两者之间在强制转换的时候会抛出java.lang.ClassCastException。而所以在默认情况下,如果目标对象没有实现任何接
口,Spring会选择CGLIB代理, 其生成的动态代理对象是目标类的子类。
上说的是默认情况下,也可以手动配置一些选项使Spring采用CGLIB代理。
org.springframework.transaction.interceptor.TransactionProxyFactoryBean
是org.springframework.aop.framework.
ProxyConfig的子类,所以可以参照ProxyConfig里的一些设置如下所示,将optimize和proxyTargetClass任意一
个设置为true都可以强制Spring采用CGLIB代理。
如果当需要使用CGLIB代理和@AspectJ自动代理支持,请按照如下的方式设置 |aop:aspectj-autoproxy| 的 proxy-target-class 属性
:
|aop:aspectj-autoproxy proxy-target-class="true"/|
这样使用CGLIB代理也就不会出现前面提到的ClassCastException问题了,也可以在性能上有所提高,关键是对于代理对象是否继承接口可以统一使用。
转载自:http://blog.csdn.net/WWWTYB/article/details/5703868
- spring 动态代理如何决定使用jdk代理和cglib(网易面试题)
- spring 如何决定使用jdk动态代理和cglib(网易面试题)
- spring 如何决定使用jdk动态代理和cglib(网易面试题)
- jdk动态代理与cglib代理、spring Aop代理原理-代理使用浅析
- 重温Spring之旅5——AOP代理对象、JDK动态代理、使用cglib生产代理
- 基于JDK动态代理和CGLIB动态代理的实现Spring注解管理事务(@Trasactional)到底有什么区别
- 记录spring使用JDK动态代理,发生的2次代理问题
- 基于JDK动态代理和CGLIB动态代理的实现Spring注解管理事务
- Spring—AOP两种代理机制对比(JDK和CGLib动态代理)
- Spring系列之 (十一):AOP实现方式:动态代理的两个方式(JDK和Cglib)
- Spring AOP中的JDK和CGLib动态代理哪个效率更高?
- Spring(十)通过动态代理(JDK的Proxy)和cglib实现AOP技术
- 静态代理、JDK动态代理、CGLIB动态代理、Spring实现AOP、IOC+AOP
- spring对AOP的支持(JDK的动态代理实现AOP和CGLIB实现AOP)
- JDK动态代理和CGLIB动态代理,实现Spring注解管理事务区别。
- Spring学习总结(二)——静态代理、JDK与CGLIB动态代理、AOP+IoC
- Sring如何选择JDK动态代理与CGLIB字节码增强
- Spring -- tx:annotation-driven 注解基于JDK动态代理和CGLIB动态代理的实现Spring注解管理事务(@Trasactional)的区别
- 使用JDK和Cglib两种方式动态代理
- Spring: proxy-target-class 决定 用 CGlib 还是 JDK AOP 来生成代理