• 极客专栏正式上线!欢迎访问 https://www.jikewenku.com/topic.html
  • 极客专栏正式上线!欢迎访问 https://www.jikewenku.com/topic.html

spring之事务回滚技巧

技术杂谈 勤劳的小蚂蚁 4个月前 (12-29) 85次浏览 已收录 0个评论 扫描二维码


1、遇到的问题
当我们一个方法里面有多个数据库保存操作的时候,中间的数据库操作发生的错误。伪代码如下:
publicmethod() {
   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、实例
这里使用的事务配置如下:
<!– Jpa 事务配置 –>
   <beanid=“transactionManager”class=“org.springframework.orm.jpa.JpaTransactionManager”>
       <propertyname=“entityManagerFactory”ref=“entityManagerFactory”/>
   </bean>
   
   <!– 开启注解事务 –>
   <tx:annotation-driventransaction-manager=“transactionManager”proxy-target-class=“true” />

在spring的配置文件中,如果数据源的defaultAutoCommit设置为True了,那么方法中如果自己捕获了异常,事务是不会回滚的,如果没有自己捕获异常则事务会回滚,如下例

比如配置文件里有这么条记录
<bean id=“dataSource”class=“com.alibaba.druid.pool.DruidDataSource”>

<property name=“xxx”value=“xxx”/>

<property name=“xxx”value=“xxx”/>

….
<property name=“defaultAutoCommit”value=“true” />

</bean>

可能你会发现你并没有配置这个参数,是不是他就不会自动提交呢?答案是不是的,我这里是使用了com.alibaba.druid.pool.DruidDataSource作为数据库连接池,默认的defaultAutoCommit就是true,可以看下面的源码

那么现在有两个情况
  
情况1:如果没有在程序中手动捕获异常
@Transactional(rollbackOn = { Exception.class })  
publicvoidtest()throws Exception {  
    doDbStuff1();  
    doDbStuff2();
//假如这个操作数据库的方法会抛出异常,
//现在方法doDbStuff1()对数据库的操作   会回滚。  
}

情况2:如果在程序中自己捕获了异常
@Transactional(rollbackOn = { Exception.class })  
publicvoidtest(){  
    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()是会回滚的  
   }  
}




丨极客文库, 版权所有丨如未注明 , 均为原创丨
本网站采用知识共享署名-非商业性使用-相同方式共享 3.0 中国大陆许可协议进行授权
转载请注明原文链接:spring之事务回滚技巧
喜欢 (0)
[247507792@qq.com]
分享 (0)
勤劳的小蚂蚁
关于作者:
温馨提示:本文来源于网络,转载文章皆标明了出处,如果您发现侵权文章,请及时向站长反馈删除。

您必须 登录 才能发表评论!

  • 精品技术教程
  • 编程资源分享
  • 问答交流社区
  • 极客文库知识库

客服QQ


QQ:2248886839


工作时间:09:00-23:00