springboot 注解@Transactional失效的原因

参考源码分析

https://blog.csdn.net/oyc619491800/article/details/105826597

一:声明式事务@Transactional
在这里插入图片描述
二:注解@Transactional使用
在这里插入图片描述
三:常用,使用该注解时,记得加异常回滚,
在这里插入图片描述
readOnly 属性

  • readOnly :指定事务是否为只读事务,默认值为 false;为了忽略那些不需要事务的方法,比如读取数据,可以设置
    read-only 为 true。

rollbackFor 属性

  • rollbackFor :用于指定能够触发事务回滚的异常类型,可以指定多个异常类型。

不加异常回归的话,自定义异常失败,但不回滚;
在这里插入图片描述
四:使用该注解注意事项以及失效情况

总结:
三种情况失效最多是自身调用、异常被吃、异常抛出类型不对这三个了。

1、@Transactional 应用在非 public 修饰的方法上

如果Transactional注解应用在非public修饰的方法上,Transactional将会失效。

  • 注意:protected、private修饰的方法上使用 @Transactional
    注解,虽然事务无效,但不会有任何报错,这是我们很容犯错的一点。

2、@Transactional 注解属性 propagation 设置错误

这种失效是由于配置错误,若是错误的配置以下三种 propagation,事务将不会发生回滚。

  • TransactionDefinition.PROPAGATION_SUPPORTS:如果当前存在事务,则加入该事务;如果当前没有事务,则以非事务的方式继续运行。
  • TransactionDefinition.PROPAGATION_NOT_SUPPORTED:以非事务方式运行,如果当前存在事务,则把当前事务挂起。
  • TransactionDefinition.PROPAGATION_NEVER:以非事务方式运行,如果当前存在事务,则抛出异常。

3、@Transactional 注解属性 rollbackFor 设置错误

rollbackFor 可以指定能够触发事务回滚的异常类型。Spring默认抛出了未检查unchecked异常(继承自
RuntimeException 的异常)或者 Error才回滚事务;其他异常不会触发回滚事务。如果在事务中抛出其他类型的异常,但却期望
Spring 能够回滚事务,就需要指定 rollbackFor属性。

在这里插入图片描述
4、同一个类中方法调用,导致@Transactional失效

对同一个类里面的方法调用,比如有一个类Test,它的一个方法A,A再调用本类的方法B(不论方法B是用public还是private修饰),但方法A没有声明注解事务,而B方法有。则外部调用方法A之后,方法B的事务是不会起作用的。

update方法上面没有加 @Transactional 注解,调用有 @Transactional 注解的 updateOrder 方法,updateOrder 方法上的事务不管用

public class OrderServiceImpl implements OrderService {

 
    public void update(Order order) {

        updateOrder(order);

    }

 
    @Transactional

    public void updateOrder(Order order) {

        // update order

    }

 
}

5、异常被你的 catch“吃了”导致@Transactional失效

把异常吃了,然后又不抛出来,事务怎么回滚吧!

在业务方法中一般不需要catch异常,如果非要catch一定要抛出throw new RuntimeException(),或者注解中指定抛异常类型@Transactional(rollbackFor=Exception.class),否则会导致事务失效,数据commit造成数据不一致,所以有些时候try catch反倒会画蛇添足。

// @Service

public class OrderServiceImpl implements OrderService {

 
    @Transactional

    public void updateOrder(Order order) {

        try {

            // update order

        } catch {

 
        }

    }

 
}

6、数据库引擎不支持事务

这种情况出现的概率并不高,事务能否生效数据库引擎是否支持事务是关键。常用的MySQL数据库默认使用支持事务的innodb引擎。一旦数据库引擎切换成不支持事务的myisam,那事务就从根本上失效了。

  1. 异常类型错误

这样事务也是不生效的,因为默认回滚的是:RuntimeException,如果你想触发其他异常的回滚,需要在注解上配置一下,如:

@Transactional(rollbackFor = Exception.class)

// @Service

public class OrderServiceImpl implements OrderService {

 
    @Transactional

    public void updateOrder(Order order) {

        try {

            // update order

        } catch {

            throw new Exception("更新错误");

        }

    }

 
}
  1. 不支持事务

Propagation.NOT_SUPPORTED: 表示不以事务运行,当前若存在事务则挂起,都主动不支持以事务方式运行了,那事务生效也是白搭!

@Service

public class OrderServiceImpl implements OrderService {

 
    @Transactional

    public void update(Order order) {

        updateOrder(order);

    }

 
    @Transactional(propagation = Propagation.NOT_SUPPORTED)

    public void updateOrder(Order order) {

        // update order

    }

 
}
Logo

为开发者提供学习成长、分享交流、生态实践、资源工具等服务,帮助开发者快速成长。

更多推荐