在 Spring Boot 中,造成事务不自动回滚的场景有很多,比如以下这些:
- 非 public 修饰的方法中的事务不自动回滚;
- 当 @Transactional 遇上 try/catch 事务不自动回滚;
- 调用类内部的 @Transactional 方法事务不自动回滚;
- 抛出检查异常时事务不自动回滚;
- 数据库不支持事务,事务也不会自动回滚。
那么对于上面的这些场景,我们应该如何解决呢?接下来我们一一来看。
1.非 public 方法解决方案
非 public 方法中事务不回滚的直接原因是,在非 public 方法上添加的 @Transactional 关键字是无效的,也就是此方法本身是以非事务的方式运行的,所以它当然不会自动回滚事务了。
因为 @Transactional 使用的是 Spring AOP 实现的,而 Spring AOP 是通过动态代理实现的,而 @Transactional 在生成代理时会判断,如果方法为非 public 修饰的方法,则不生成代理对象,这样也就没办法自动回滚事务了,它的部分实现源码如下:
protected TransactionAttribute computeTransactionAttribute(Method method, Class<?> targetClass) {
// Don't allow no-public methods as required.
// 非 public 方法,设置为 null
if (allowPublicMethodsOnly() &&