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());
}
}
}
说明:
让事务起作用,遇到错误进行回滚,应该注意的事项:
- 第一种情况:同一个类中 一个方法无嵌套方法
- 如果方法名上加上@Transactional注解,方法内不要用try catch ;如果必须要用try catch ,则catch中必须用throw new RuntimeException()。否则事务不起作用。
- 第二种情况:同一个类中 方法A嵌套方法B
- 方法A有@Transactional,方法内都没有try catch,事务起作用。
- 方法A有@Transactional和try catch,并且catch中用throw new RuntimeException(),事务起作用。
- 第三种情况:不同类中,方法C嵌套方法B
- 方法B上加上@Transactional注解,方法内不要用try catch ;如果必须要用try catch ,则catch中必须用throw new RuntimeException()。否则方法B的事务不起作用。
- 方法C上加上@Transactional注解,方法内不要用try catch ;如果必须要用try catch ,则catch中必须用throw new RuntimeException(),此时方法B怎么写都行。否则方法C的事务不起作用。
结语
亲测是有用的,总结起来就是:
-
要想事务起作用,必须是主方法名上有@Transactional注解,方法体内不能用try catch;如果用try catch,则catch中必须用throw new Exception();
-
@Transactional注解应该只被应用到public方法上,不要用在protected、private等方法上,即使用了也将被忽略,不起作用。这是由Spring AOP决定的。
-
只有来自外部的方法调用才会呗AOP代理捕捉,类内部方法调用类内部的其他方法,子方法并会不引起事务行为,即使被调用的方法上使用有@Transactional注解。
-
类内部方法调用内部的其他方法,被调用的方法体中如果有try catch,则catch中必须用throw new Exception(),否则即使主方法上加上@Transactional注解,如果被调用的子方法出错也不会抛出异常,不会引起事务起作用。