Spring AOP 自调用 (@Transactional 失效)

本文探讨了在Spring AOP中遇到的一个常见问题:当类内部方法互相调用时,@Transactional注解可能导致事务失效。通过分析问题定位,指出问题在于自调用未经过代理类,从而跳过了事务管理。文章提供了两种解决方案,包括将事务注解置于最外层或在类级别,以及利用Spring获取代理对象进行内部调用。

经常使用Spring AOP 做切面, 但是会遇到一个问题. 当在同一个类中, 调用当前类下其他方法时候, 会导致切面失效. 即:类自调用问题

 

问题描述:

例如:  B()调用 A() 方法. 按照第一直觉, B()中调了A() 之后, 因为A()有事务注解, 事务是会生效的. 但实际不会生效.

public class OrderService{
        
        @Transactional
        public void methodA(){
            
        }

        public void methodB(){
            methodA();
        }
    }

Spring 生成代理类. 大致如下. 

public class OrderServiceProxyXxxx{

        OrderService orderService;        

        public void methodA(){
            beginTransaction();
            orderService.methodA();
            endTransaction();
        }

        public void methodB(){
            orderService.methodB();
        }
    }

 问题定位:

1. 先调用B()方法. 再B()中调用A():

    由于沒有增加@Transactional. 代理类中实际是直接调用目标类中的方法. 目标类中B() 再调用目标类A(). 

    调用顺序: OrderServiceProxyXxxx.methodB() > orderSerivce.methodB() > orderSerivce.methodA()

    这个过程中可以看到. 实际沒有事务参与其中.

 2. 反过来,先调用A(). 再A()中 调用B().

       调用顺序:OrderServiceProxyXxxx.methodA() > orderSerivce.methodA() > orderSerivce.methodB()

     这里由于OrderServiceProxyXxxx.MethodA() 织入了事务, 因此事务是有效的.

 

解决方法:

 1. 不要再内部自调用方法中使用AOP.   如果是事务这种注解, 则放到最外层. 或者标记到类上.

      这里还需要注意代理方式, 是JDK还是 cglib的方式. jdk 是基于接口的方式.对非public是无效的. 需要注意下.springboot2之后默认是cglib

 2. 内部自调用, 使用代理对象调用.

       1. 方法1: 使用@Autowired 注入自身对象. 有内部方法调用时, 通过注入的代理类调用. 而不是this 调用内部方法.

 public class OrderService{
        @Autowired OrderService orderService;
        @Transactional
        public void methodA(){

        }
        public void methodB(){
            orderService.methodA();
        }
    }

       2. 方法2:使用Spring 获取代理对象

 public class OrderService{
        @Transactional
        public void methodA(){

        }
        public void methodB(){
            ((OrderService) AopContext.currentProxy()).methodA();
        }
    }

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值