Spring 事务嵌套无效

Spring Boot 关于嵌套事务无效解决方案

Spring Boot自带事务注解 @Transactional, 只要在方法或者类上加上这个注解,就可以开启事务回滚。当然,是在单个事务的情况下,内嵌事务的时候却不起了作用,如方法A加了这个注解,方法B也加了注解,方法A里调用了方法B,那方法A出现了错误,而方法B没有,那方法B不会进行回滚

代码示例

方法A的代码

@RestController
public class ServiceA{
    @Autowired
    private ServiceB serviceB;
    @Autowired
    private TestMapper mapper;

    @Transactional(rollbackFor = Exception.class)
    @PostMapping("/method1")
    public void method1(){
        try {
            serviceB.insert(new Test());
            mapper.findById(1);
        }catch (Exception e){
            e.printStackTrace();
        }
    }
}    

方法B的代码

@Serivce
public class ServiceB{
    @Autowired
    private TestMapper mapper;

    @Transactional(rollbackFor = Exception.class)
    public void insert(Test test) throws Exception{
        mapper.insert(test);
    }
}   

在这个方法中(method1),这个方法有带了回滚注解,serviceB.insert方法里面也带了注解,那么有这么一种情况,在findById出现异常了之后,insert的方法会被回滚吗?

我试了一下,不会回滚,就算findById出了问题,前面的insert还是执行了,原因是因为事务内嵌,最好不要被try catch,外事务被try catch之后,事务就会失效了,最好是异常都抛出去,真的需要被try catch的时候,在catch代码块里,抛一个异常出去

修改后的代码

@RestController
public class ServiceA{
    @Autowired
    private ServiceB serviceB;
    @Autowired
    private TestMapper mapper;

    @Transactional(rollbackFor = Exception.class)
    @PostMapping("/method1")
    public void method1(){
        try {
            serviceB.insert(new Test());
            mapper.findById(1);
        }catch (Exception e){
            e.printStackTrace();
            //抛出异常
            throw new RuntimeException(e.getMessage());
        }
    }
}   

说明:
让事务起作用,遇到错误进行回滚,应该注意的事项:

  • 第一种情况:同一个类中 一个方法无嵌套方法
  1. 如果方法名上加上@Transactional注解,方法内不要用try catch ;如果必须要用try catch ,则catch中必须用throw new RuntimeException()。否则事务不起作用。
  • 第二种情况:同一个类中 方法A嵌套方法B
  1. 方法A有@Transactional,方法内都没有try catch,事务起作用。
  2. 方法A有@Transactional和try catch,并且catch中用throw new RuntimeException(),事务起作用。
  • 第三种情况:不同类中,方法C嵌套方法B
  1. 方法B上加上@Transactional注解,方法内不要用try catch ;如果必须要用try catch ,则catch中必须用throw new RuntimeException()。否则方法B的事务不起作用。
  2. 方法C上加上@Transactional注解,方法内不要用try catch ;如果必须要用try catch ,则catch中必须用throw new RuntimeException(),此时方法B怎么写都行。否则方法C的事务不起作用。

结语

亲测是有用的,总结起来就是:

  1. 要想事务起作用,必须是主方法名上有@Transactional注解,方法体内不能用try catch;如果用try catch,则catch中必须用throw new Exception();

  2. @Transactional注解应该只被应用到public方法上,不要用在protected、private等方法上,即使用了也将被忽略,不起作用。这是由Spring AOP决定的。

  3. 只有来自外部的方法调用才会呗AOP代理捕捉,类内部方法调用类内部的其他方法,子方法并会不引起事务行为,即使被调用的方法上使用有@Transactional注解。

  4. 类内部方法调用内部的其他方法,被调用的方法体中如果有try catch,则catch中必须用throw new Exception(),否则即使主方法上加上@Transactional注解,如果被调用的子方法出错也不会抛出异常,不会引起事务起作用。

你可能感兴趣的:(Spring,Boot)