Spring AOP 自调用 (@Transactional 失效)

经常使用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();
        }
    }

 

你可能感兴趣的:(Spring,SpringAOP,自调用,Spring事务失效)