@Transactional注解的rollbackFor属性

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了@Transactional注解的rollbackFor属性相关的知识,希望对你有一定的参考价值。

参考技术A 先回忆下java的异常模型,Throwable是最顶层的父类,有Error和Exception两个子类。
Error表示严重的错误(如OOM等);
Exception可以分为运行时异常(RuntimeException及其子类)和非运行时异常(Exception的子类中,除了RuntimeException及其子类之外的类)。

非运行时异常是检查异常(checked exceptions),一定要try catch,因为这类异常是可预料的,编译阶段就检查的出来;
Error和运行时异常是非检查异常(unchecked exceptions),不需要try catch,因为这类异常是不可预料的,编辑阶段不会检查, 没必要检查,也检查不出来 。

spring的@Transactional注解可以很方便的开启事务,但是 默认只在遇到运行时异常和Error时才会回滚,非运行时异常不回滚 ,即Exception的子类中,除了RuntimeException及其子类,其他的类默认不回滚(不知道为什么要这样设计?)
而rollbackFor属性可以解决这个问题, rollbackFor = Exception.class表示Exception及其子类的异常都会触发回滚,同时不影响Error的回滚。

下面是关于@Transactional注解的一些实验

实验一
不加rollbackFor属性,抛出RuntimeException,正常回滚

实验二
不加rollbackFor属性,抛出IOException,不回滚

实验三
加上rollbackFor = Exception.class,抛出IOException,正常回滚

实验四
不加rollbackFor属性,抛出OutOfMemoryError,正常回滚

实验五
加上rollbackFor = Exception.class,抛出OutOfMemoryError,正常回滚, 说明rollbackFor = Exception.class不会覆盖Error的回滚

Springboot 事务回滚不生效可能出现的原因

检查一下几点:

1.数据库是否为InnoDB引擎,具体原因请查看 Mysql引擎类型刨析

2.配置文件中是否开启

transaction:
rollback-on-commit-failure: true

3.是否在启动类中加入注解(此项可忽略,spring boot 默认开启了事务)

@EnableTransactionManagement

4.@Transactional注解位置:

@Transactional注解必须和抛出异常的位置在一起

在Service中加入的事务注解,手动抛出异常时要在Service中抛出,才能看到效果

如果Controller中,调用两个不同Service的方法并开启了事务回滚,要想事务生效,则需要在Controller也加入@Transactional注解

5.@Transactional注解默认只能拦截RuntimeException和Error,如果自定义的Exception,需要设置rollbackFor 属性值,如下

@Transactional(rollbackFor = Exception.class)

注意:添加@Transactional的方法必须是public

以上是关于@Transactional注解的rollbackFor属性的主要内容,如果未能解决你的问题,请参考以下文章

Spring中@Transactional事务回滚

Springboot 事务回滚不生效可能出现的原因

2017.5.27 使用propagation实现:根据参数决定是否需要事务管理

@EnableTransactionManagement以及 @Transactional 注解

事务注解@Transactional不起作用

spring @Transactional 事务注解的坑