项目中写了一个关务事务的方法,然后发现不生效。然后我做了下面的一个测试看看到底哪里出了问题!
这是我的简化后的测试代码,上面有个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"/>
<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、事务的传播模式