Spring事务@Transactional(rollbackFor = Exception.class) 不生效

Posted 老周聊架构

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Spring事务@Transactional(rollbackFor = Exception.class) 不生效相关的知识,希望对你有一定的参考价值。

欢迎大家关注我的公众号【老周聊架构】,Java后端主流技术栈的原理、源码分析、架构以及各种互联网高并发、高性能、高可用的解决方案。

一、发现问题

项目中写了一个关务事务的方法,然后发现不生效。然后我做了下面的一个测试看看到底哪里出了问题!

这是我的简化后的测试代码,上面有个insert操作数据库的插入方法,下面我就直接抛个异常,省得写测试操作数据库不成功的另一个方法。我们都知道事务是作为一个整体来执行,包含在其中的对数据库的操作要么全部执行,要么都不执行。所以我下面来演示一个成功插入,另一个报错的情况。这种情况加了事务只要有一个是异常,那么这都不能插入数据。

我们来看下面的例子:

@Transactional(rollbackFor = Exception.class)
private void insertUser(UserDomain user) 
    user.setUserId(3);
    user.setUserName("riemann");
    user.setPassword("root");
    user.setPhone("13129535588");
    userDao.insert(user);
    throw new RuntimeException("private test");

然后我们执行后,神奇的发现
异常正常打印了

而数据库却成功插入了。

二、解决问题

经查阅资料发现,原来这是 事务@Transactional 的可见性。

1、@Transactional 注解只应用到 public 修饰的方法上,在 protected、private 修饰的方法上都不会起作用!(事务具有可见性)

2、一个类中假设 方法A 使用了注解 @Transactional ,同一个类中的 方法B 再去调用方法A 时,事务不生效!(事务具有传递性)

上面的问题不用我说了吧,直接把 private 改成 public ,事务就生效了。

我们来看事务具有传递性的也就是第2点。例子如下:

public ReturnT transactionTest(UserDomain user) 
    insertUser(user);
    return new ReturnT("success");


@Transactional(rollbackFor = Exception.class)
public void insertUser(UserDomain user) 
    user.setUserId(3);
    user.setUserName("riemann");
    user.setPassword("root");
    user.setPhone("13129535588");
    userDao.insert(user);
    int i = 1 / 0;
    System.out.println(i);

这个例子恰恰说明了第2点事务不生效。

那我们怎么解决呢?把事务加到上一层的方法中去就可以了,因为事务具有传递性。

@Transactional(rollbackFor = Exception.class)
public ReturnT transactionTest(UserDomain user) 
    insertUser(user);
    return new ReturnT("success");


public void insertUser(UserDomain user) 

    user.setUserId(3);
    user.setUserName("riemann");
    user.setPassword("root");
    user.setPhone("13129535588");
    userDao.insert(user);
    int i = 1 / 0;
    System.out.println(i);

三、举一反三

后面再遇到了事务不生效的情况,可以从下面几点找原因:

1、检查你的方法是不是 public 修饰的。
2、检查是不是同一个类中的方法调用(如a方法调用同一个类中的b方法,在b方法上加的事务)。
3、你的异常类型是不是unchecked异常?如果我想check异常也想回滚怎么办,注解上面写明异常类型即可。

@Transactional(rollbackFor=Exception.class) 

类似的还有 norollbackFor,自定义不回滚的异常

4、查看自己的数据的引擎,如果是 mysql,注意表要使用支持事务的引擎,比如innodb,如果是myisam,事务是不起作用的。

查看 MySQL 的所有存储引擎:show engines;
查看 MySQL 的当前存储引擎:show variables like '%storage_engine%';

5、异常是不是被你catch住了

6、如果你是 ssm 框架,在用 xml 配置开启的事务,那你要看一下:

  • 是否开启了对注解的解析
<tx:annotation-driven transaction-manager="transactionManager" proxy-target-class="true"/>
  • spring是否扫描到你这个包,如下是扫描到org.test下面的包
<context:component-scan base-package="org.test" ></context:component-scan>

四、知识拓展

1、默认情况下,Spring 会对 unchecked 异常进行事务回滚;如果是 checked 异常则不回滚。
那么什么是 checked 异常,什么是 unchecked 异常呢?

java里面将派生于Error或者RuntimeException(比如空指针,1/0)的异常称为unchecked异常,其他继承自java.lang.Exception得异常统称为Checked Exception,如IOException、TimeoutException等

2、只读事务

@Transactional(propagation=Propagation.NOT_SUPPORTED,readOnly=true)

只读标志只在事务启动时应用,否则即使配置也会被忽略。
启动事务会增加线程开销,数据库因共享读取而锁定(具体跟数据库类型和事务隔离级别有关)。通常情况下,仅是读取数据时,不必设置只读事务而增加额外的系统开销。

3、编程式事务与声明式事务的区别

关于编程式事务与声明式事务的区别可以看这一篇博文:

SpringBoot项目中编程式事务与声明式事务的区别?

4、数据库隔离级别有哪些

这一篇关于 MySQL的四种隔离级别 讲的很清楚了:

数据库隔离级别有哪些,各自的含义是什么,MYSQL默认的隔离级别是是什么?

5、事务的传播模式

  • REQUIRED(默认模式):业务方法需要在一个容器里运行。如果方法运行时,已经处在一个事务中,那么加入到这个事务,否则自己新建一个新的事务。
  • NOT_SUPPORTED:声明方法不需要事务。如果方法没有关联到一个事务,容器不会为他开启事务,如果方法在一个事务中被调用,该事务会被挂起,调用结束后,原先的事务会恢复执行。
  • REQUIRESNEW:不管是否存在事务,该方法总汇为自己发起一个新的事务。如果方法已经运行在一个事务中,则原有事务挂起,新的事务被创建。
  • MANDATORY:该方法只能在一个已经存在的事务中执行,业务方法不能发起自己的事务。如果在没有事务的环境下被调用,容器抛出例外。
  • SUPPORTS:该方法在某个事务范围内被调用,则方法成为该事务的一部分。如果方法在该事务范围外被调用,该方法就在没有事务的环境下执行。
  • NEVER:该方法绝对不能在事务范围内执行。如果在就抛例外。只有该方法没有关联到任何事务,才正常执行。
  • NESTED:如果一个活动的事务存在,则运行在一个嵌套的事务中。如果没有活动事务,则按REQUIRED属性执行。它使用了一个单独的事务,这个事务拥有多个可以回滚的保存点。内部事务的回滚不会对外部事务造成影响。它只对DataSourceTransactionManager事务管理器起效。

以上是关于Spring事务@Transactional(rollbackFor = Exception.class) 不生效的主要内容,如果未能解决你的问题,请参考以下文章

玩转Spring--消失的事务@Transactional

使用 Spring @Transactional 管理 Hibernate 事务

Spring Boot事务和事务传播机制

Spring中@Transactional事务回滚

Spring源码剖析-Transactional 事务执行流程

Spring 下默认事务机制中@Transactional 无效的原因