2023年11月29日发(作者:)

spring事务回滚

1

当我们⼀个⽅法⾥⾯有多个数据库保存操作的时候,中间的数据库操作发⽣的错误。伪代码如下:

public method() {

(Person1);

(Person2);

(Person2);//假如这句发⽣了错误,前⾯的两个对象会被保存到数据库中

(Person2);

}

期待的情况:发⽣错误之前的所有数据库保存操作都回滚,即不保存

正常情况:前⾯的数据库操作会被执⾏,⽽发⽣数据库操作错误开始及之后的所有的数据保存操作都将失败。这样⼦应该都不是我们要的结果吧。

当遇到这种情况,我们就可以使⽤Spring的事务解决这个问题。

2

1) 异常的架构

异常的继承结构:Throwable为基类,ErrorException继承ThrowableRuntimeExceptionIOException等继承ExceptionErrorRuntimeException及其⼦类成为未检

查异常(unchecked),其它异常成为已检查异常(checked)。

2Error异常

Error表⽰程序在运⾏期间出现了⼗分严重、不可恢复的错误,在这种情况下应⽤程序只能中⽌运⾏,例如JAVA 虚拟机出现错误。Error是⼀种unchecked Exception,编译

器不会检查Error是否被处理,在程序中不⽤捕获Error类型的异常。⼀般情况下,在程序中也不应该抛出Error类型的异常。

3RuntimeException异常

Exception异常包括RuntimeException异常和其他⾮RuntimeException的异常。

RuntimeException 是⼀种Unchecked Exception,即表⽰编译器不会检查程序是否对RuntimeException作了处理,在程序中不必捕获RuntimException类型的异常,也不必

在⽅法体声明抛出 RuntimeException类。RuntimeException发⽣的时候,表⽰程序中出现了编程错误,所以应该找出错误修改程序,⽽不是去捕获RuntimeException

4Checked Exception异常

Checked Exception异常,这也是在编程中使⽤最多的Exception,所有继承⾃Exception并且不是RuntimeException的异常都是checked Exception,上图中的IOException

ClassNotFoundExceptionJAVA 语⾔规定必须对checked Exception作处理,编译器会对此作检查,要么在⽅法体中声明抛出checked Exception,要么使⽤catch语句捕获

checked Exception进⾏处理,不然不能通过编译。

3、实例

这⾥使⽤的事务配置如下:

<bean id="transactionManager" class="nsactionManager">

<property name="entityManagerFactory" ref="entityManagerFactory"/>

bean>

<tx:annotation-driven transaction-manager="transactionManager" proxy-target-class="true" />

spring的配置⽂件中,如果数据源的defaultAutoCommit设置为True了,那么⽅法中如果⾃⼰捕获了异常,事务是不会回滚的,如果没有⾃⼰捕获异常则事务会回滚,如下

⽐如配置⽂件⾥有这么条记录

<bean id="dataSource" class="ataSource">

<property name="xxx" value="xxx"/>

<property name="xxx" value="xxx"/>

....

<property name="defaultAutoCommit" value="true" />

bean>

可能你会发现你并没有配置这个参数,是不是他就不会⾃动提交呢?答案是不是的,我这⾥是使⽤了ataSource作为数据库连接池,默认的

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

tackTrace();

}

}

现在如果我们需要⼿动捕获异常,并且也希望抛异常的时候能回滚肿么办呢?

下⾯这样写就好了,⼿动回滚事务:

@Transactional(rollbackOn = { Exception.class })

public void test() {

try {

doDbStuff1();

doDbStuff2();

} catch (Exception e) {

tackTrace();

tTransactionStatus().setRollbackOnly();//就是这⼀句了,加上之后,如果doDbStuff2()抛了异常, //doDbStuff1()是会回滚的

}

}

致谢:感谢您的阅读!转⽂请加原⽂链接,谢谢!