您好,欢迎访问一九零五行业门户网

spring 事务回滚

1、遇到的问题
当我们一个方法里面有多个数据库保存操作的时候,中间的数据库操作发生的错误。伪代码如下:
public method() { dao1.save(person1); dao1.save(person2); dao1.save(person2);//假如这句发生了错误,前面的两个对象会被保存到数据库中 dao1.save(person2); }
期待的情况:发生错误之前的所有数据库保存操作都回滚,即不保存 正常情况:前面的数据库操作会被执行,而发生数据库操作错误开始及之后的所有的数据保存操作都将失败。这样子应该都不是我们要的结果吧。 当遇到这种情况,我们就可以使用spring的事务解决这个问题。 2、异常的一些基本知识 1) 异常的架构 异常的继承结构:throwable为基类,error和exception继承throwable,runtimeexception和ioexception等继承exception。error和runtimeexception及其子类成为未检查异常(unchecked),其它异常成为已检查异常(checked)。
2)error异常
error表示程序在运行期间出现了十分严重、不可恢复的错误,在这种情况下应用程序只能中止运行,例如java 虚拟机出现错误。error是一种unchecked exception,编译器不会检查error是否被处理,在程序中不用捕获error类型的异常。一般情况下,在程序中也不应该抛出error类型的异常。
3)runtimeexception异常
exception异常包括runtimeexception异常和其他非runtimeexception的异常。
runtimeexception 是一种unchecked exception,即表示编译器不会检查程序是否对runtimeexception作了处理,在程序中不必捕获runtimexception类型的异常,也不必在方法体声明抛出 runtimeexception类。runtimeexception发生的时候,表示程序中出现了编程错误,所以应该找出错误修改程序,而不是去捕获runtimeexception。
4)checked exception异常
checked exception异常,这也是在编程中使用最多的exception,所有继承自exception并且不是runtimeexception的异常都是checked exception,上图中的ioexception和classnotfoundexception。java 语言规定必须对checked exception作处理,编译器会对此作检查,要么在方法体中声明抛出checked exception,要么使用catch语句捕获checked exception进行处理,不然不能通过编译。
3、实例
这里使用的事务配置如下:

在spring的配置文件中,如果数据源的defaultautocommit设置为true了,那么方法中如果自己捕获了异常,事务是不会回滚的,如果没有自己捕获异常则事务会回滚,如下例
比如配置文件里有这么条记录
....
可能你会发现你并没有配置这个参数,是不是他就不会自动提交呢?答案是不是的,我这里是使用了com.alibaba.druid.pool.druiddatasource作为数据库连接池,默认的defaultautocommit就是true,可以看下面的源码
那么现在有两个情况
情况1:如果没有在程序中手动捕获异常
@transactional(rollbackon = { exception.class }) public void test() throws exception { dodbstuff1(); dodbstuff2();//假如这个操作数据库的方法会抛出异常,现在方法dodbstuff1()对数据库的操作 会回滚。 }
情况2:如果在程序中自己捕获了异常
@transactional(rollbackon = { exception.class }) public void test() { try { dodbstuff1(); dodbstuff2();//假如这个操作数据库的方法会抛出异常,现在方法dodbstuff1()对数据库的操作 不会回滚。 } catch (exception e) { e.printstacktrace(); } }
现在如果我们需要手动捕获异常,并且也希望抛异常的时候能回滚肿么办呢?
下面这样写就好了,手动回滚事务:
@transactional(rollbackon = { exception.class }) public void test() { try { dodbstuff1(); dodbstuff2(); } catch (exception e) { e.printstacktrace(); transactionaspectsupport.currenttransactionstatus().setrollbackonly();//就是这一句了,加上之后,如果dodbstuff2()抛了异常, //dodbstuff1()是会回滚的 } }
其它类似信息

推荐信息