jtlgb

导航

 

ServiceA.java文件:

 

 

查看Spring Tx的相关日志:  可以看到只创建了一个事物ServiceA.service方法的事务,但是callSelf方法却没有被事务增强;

         分析原因:Spring事务生成的对象也是被Cglib或JDK代理的对象,就区别于该对象本身了,

代理的对象执行方法之前会走拦截器链,就不能同this方法.

 

         解决方案:

  1. 使用Autowired注解将自身注入,然后调用注入属性的方法;

 

验证输出结果:  确实初始化了callSelf的事务;

 

 

2.之前Aop可以将代理对象暴露到当前线程局部变量中;

<aop:aspectj-autoproxy expose-proxy="true"/>

 

通过尝试发现,SpringTx也可以使用该配置,将创建的对象加入到当前线程局部变量;

也许觉得SpringAop和SpringTx不一样啊,但其实两者都实现了AbstractAutoProxyCreator类,同样设置expose-proxy也能生效,绑定到线程局部变量上;

调用方式如下:

 

         验证输出结果:可以看到确实 callSelf方法也被Spring事务增强到了.

 

 

3.按照道理来说tx:annotation-driven标签应该也有expose-proxy属性,但是很不幸:

<tx:annotation-driven>标签确实不允许expose-proxy属性设置,有proxy-target-class属性;

查看AopNamespaceUtils的useClassProxyingIfNecessary方法:

Line  91-94行是用来解析标签时候定义expose-proxy、proxy-target-class两个属性,Aop以及Tx都会调用这个方法, 问题就是这里了, <aop:aspectj-autoproxy>标签约束里有expose-proxy,

那可以解析,但是<tx:annotation-driven>标签没有expose-proxy我解析啥子呢,

改动下解析Tx标签时,将expose-proxy设置为true; (此处涉及到改动源代码,jar包是无法直接修改的,我修改的Spring的源码,如果需要导入Spring的jar包 然后在修改之后导出成jar包)

 

验证下输出结果: 可以看到也实现了调用自身方法, 其实和方法二应该是一个意思;

posted on 2019-07-03 10:13  jtlgb  阅读(6119)  评论(0编辑  收藏  举报