@Transactional事务回滚不生效如何排查

618 阅读3分钟

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

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

@Transactional(rollbackFor=Exception.class) 
1

类似的还有 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"/> 1

  • spring是否扫描到你这个包,如下是扫描到org.test下面的包

    <context:component-scan base-package="org.test" ></context:component-scan> 1

四、知识拓展

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)
1

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

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

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

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

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

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

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

5、事务的传播模式

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