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、编程式事务与声明式事务的区别
关于编程式事务与声明式事务的区别可以看这一篇博文:
4、数据库隔离级别有哪些
这一篇关于 MySQL的四种隔离级别 讲的很清楚了:
数据库隔离级别有哪些,各自的含义是什么,MYSQL默认的隔离级别是是什么?
5、事务的传播模式
- REQUIRED(默认模式):业务方法需要在一个容器里运行。如果方法运行时,已经处在一个事务中,那么加入到这个事务,否则自己新建一个新的事务。
- NOT_SUPPORTED:声明方法不需要事务。如果方法没有关联到一个事务,容器不会为他开启事务,如果方法在一个事务中被调用,该事务会被挂起,调用结束后,原先的事务会恢复执行。
- REQUIRESNEW:不管是否存在事务,该方法总汇为自己发起一个新的事务。如果方法已经运行在一个事务中,则原有事务挂起,新的事务被创建。
- MANDATORY:该方法只能在一个已经存在的事务中执行,业务方法不能发起自己的事务。如果在没有事务的环境下被调用,容器抛出例外。
- SUPPORTS:该方法在某个事务范围内被调用,则方法成为该事务的一部分。如果方法在该事务范围外被调用,该方法就在没有事务的环境下执行。
- NEVER:该方法绝对不能在事务范围内执行。如果在就抛例外。只有该方法没有关联到任何事务,才正常执行。
- NESTED:如果一个活动的事务存在,则运行在一个嵌套的事务中。如果没有活动事务,则按REQUIRED属性执行。它使用了一个单独的事务,这个事务拥有多个可以回滚的保存点。内部事务的回滚不会对外部事务造成影响。它只对DataSourceTransactionManager事务管理器起效。
以上是关于Spring事务@Transactional(rollbackFor = Exception.class) 不生效的主要内容,如果未能解决你的问题,请参考以下文章
使用 Spring @Transactional 管理 Hibernate 事务