spring中expose-proxy的作用与原理

 

 

 也就是说我们首先调用的是AOP代理对象而不是目标对象,首先执行事务切面,事务切面内部通过TransactionInterceptor环绕增强进行事务的增强,即进入目标方法之前开启事务,退出目标方法时提交/回滚事务

 

1、测试代码如下:

 

 1 public interface AService {  
 2     public void a();  
 3     public void b();  
 4 }  
 5    
 6 @Service()  
 7 public class AServiceImpl1 implements AService{  
 8     @Transactional(propagation = Propagation.REQUIRED)  
 9     public void a() {  
10         this.b();  
11     }  
12     @Transactional(propagation = Propagation.REQUIRES_NEW)  
13     public void b() {  
14     }  
15 }  

 

2、问题:

目标对象内部的自我调用将无法实施切面中的增强,如图所示

 

 

 此处的this指向目标对象,因此调用this.b()将不会执行b事务切面,即不会执行事务增强,因此b方法的事务定义“@Transactional(propagation = Propagation.REQUIRES_NEW)”将不会实施,即结果是b和a方法的事务定义是一样的(我们可以看到事务切面只对a方法进行了事务增强,没有对b方法进行增强)


Q1:b中的事务会不会生效?
A1:不会,a的事务会生效,b中不会有事务,因为a中调用b属于内部调用,没有通过代理,所以不会有事务产生。
Q2:如果想要b中有事务存在,要如何做?
A2:<aop:aspectj-autoproxy expose-proxy=“true”> ,设置expose-proxy属性为true,将代理暴露出来,使用AopContext.currentProxy()获取当前代理,将this.b()改为((UserService)AopContext.currentProxy()).b()

3、解决方案

此处a方法中调用b方法时,只要通过AOP代理调用b方法即可走事务切面,即可以进行事务增强,如下所示:

Java代码  
1 public void a() {  
2   aopProxy.b();//即调用AOP代理对象的b方法即可执行事务切面进行事务增强  
3 } 

判断一个Bean是否是AOP代理对象可以使用如下三种方法:

AopUtils.isAopProxy(bean)        : 是否是代理对象;

AopUtils.isCglibProxy(bean)       : 是否是CGLIB方式的代理对象;

AopUtils.isJdkDynamicProxy(bean) : 是否是JDK动态代理方式的代理对象;

4、通过ThreadLocal暴露Aop代理对象

1、开启暴露Aop代理到ThreadLocal支持(如下配置方式从spring3开始支持)

1 Java代码  
2 <aop:aspectj-autoproxy expose-proxy="true"/><!—注解风格支持-->  
3 Java代码  
4 <aop:config expose-proxy="true"><!—xml风格支持--> 

 

2、修改我们的业务实现类

this.b();-----------修改为--------->((AService) AopContext.currentProxy()).b();

posted @ 2019-08-29 17:15  沐之橙  阅读(4606)  评论(0编辑  收藏  举报