spring aop类内部调用不拦截原因及解决方案

本文探讨了Spring框架中事务管理(@Transactional)在Service层的应用,重点分析了AOP内部调用不触发事务的原因及解决方法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

  spring对应java web开发的同学来说,都不陌生,其中事务@Transactional在service层更是常常使用。

1.aop类内部调用不拦截原因

细心的同学也许早就发现当service中的某个没标注@Transactional的方法调用另一个标注了@Transactional的方法时,居然没开启事务。例如

 

  1. @Service

  2. public class UserService {

  3.  
  4. @Autowired

  5. private UserMapper userMapper;

  6.  
  7. @Transactional(propagation=Propagation.REQUIRED,isolation=Isolation.DEFAULT,readOnly=false)

  8. public void insert01(User u){

  9. this.userMapper.insert(u);

  10. throw new RuntimeException("测试插入事务");

  11. }

  12.  
  13. public void insert02(User u){

  14. this.insert01(u);

  15. throw new RuntimeException("测试插入事务");

  16. }

   当controller或其他service直接调用insert01(User u)时,事务是正常的,数据库里面的确没数据;但是如果调用的是insert02(User u)方法,异常抛出了,但是数据库里面居然有数据,说明事务不正常了。

    我们知道@Transactional其实就是一个aop代理,是一个cglib动态代理(常用的动态代理有cglib动态代理,有jdk动态代理)。

    直接调用insert01(User u)时,其实流程是这样的:

 

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

  controller或其他service调用insert02(User u)时,流程是这样的:

此处的this指向目标对象,因此调用this.b()将不会执行b事务切面,即不会执行事务增强。

 

我们通过debugger方式看看controller中的userService和insert02(xxx)中的this,会发现controller中的userService是一个代理对象

看到userService=UserService

EnhanceBySpringCGLIBEnhanceBySpringCGLIB

c2174c0b

这this明显和controller中的userSerivce不是同一个了。

 

     如果同学了解cglib(执行期进行代码植入)或aspectJ(编译期就进行代码植入,反编译后看更清晰了解,看博文)。为了让同学对上面说的代理对象有个更直观的印象,我参考aspectJ那样编译后的代码植入方式,写个UserServiceProxy.java.(spring 对aop的真实处理不是这样的,我这只是举个例子直观说明,有些同学对invoke()方法不直观)

 
  1. public class UserServiceProxy extends UserService{

  2.  
  3. //模拟目标对象实例化

  4. private UserService targetService = new UserService();

  5.  
  6. @Override

  7. public void insert01(User u) {

  8. System.out.println("开启事务.....transational starting....");

  9. targetService.insert01(u);

  10. System.out.println("结束事务.....transational end......");

  11. }

  12. @Override

  13. public void insert02(User u) {

  14. targetService.insert01(u);

  15. }

  16. }


   也就是说controller中的userSerivce实际是UserServiceProxy,那么UserServiceProxy.insert02()的时候里面调用的是目标对象UserService.insert01()了,并不是代理对象UserServiceProxy.insert01(),所以事务没有开启。

 

2.aop类内部调用拦截生效的解决方案

   2.1 方案一--从beanFactory中获取对象

   刚刚上面说到controller中的UserService是代理对象,它是从beanFactory中得来的,那么service类内调用其他方法时,也先从beanFacotry中拿出来就OK了。

  1. public void insert02(User u){

  2. getService().insert01(u);

  3. }

  4. private UserService getService(){

  5. return SpringContextUtil.getBean(this.getClass());

  6. }

  2.2 方案二--获取代理对象

 

  1. private UserService getService(){

  2. // 采取这种方式的话,

  3. //@EnableAspectJAutoProxy(exposeProxy=true,proxyTargetClass=true)

  4. //必须设置为true

  5. return AopContext.currentProxy() != null ? (UserService)AopContext.currentProxy() : this;

  6. }

  如果aop是使用注解的话,那需要@EnableAspectJAutoProxy(exposeProxy=true,proxyTargetClass=true),如果是xml配置的,把expose-proxy设置为true,如

 

  1. <aop:config expose-proxy="true">

  2.        <aop:aspect ref="XXX">

  3.           <!-- 省略--->

  4.        </aop:aspect>

  5. </aop:config>

  2.3方案三--将项目转为aspectJ项目

   将项目转为aspectJ项目,aop转为aspect 类。具体就不写了

  2.4 方案四--BeanPostProcessor

   通过BeanPostProcessor 在目标对象中注入代理对象,定义InjectBeanSelfProcessor类,实现BeanPostProcessor。也不具体写了

   

   也许还有其他方案。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值