你的位置:首页 > Java教程

[Java教程]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、实例

  这里使用的事务配置如下:

 <!-- Jpa 事务配置 -->  <bean id="transactionManager" class="org.springframework.orm.jpa.JpaTransactionManager">    <property name="entityManagerFactory" ref="entityManagerFactory"/>  </bean>    <!-- 开启注解事务 -->  <tx:annotation-driven transaction-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 }) 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()是会回滚的    } } 

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




法国旅游团报价优惠法国蜜月旅游行程参考法国特价旅游多少钱到法国游费用报价大全法国自助游攻略2015丹霞山性博物馆在哪里?韶关丹霞山性博物馆乘车路线? 丹霞山性博物馆几点开门?韶关丹霞山性博物馆游玩多长时间? 2015三亚龙抬头节在哪里举办?海南三亚二月二龙抬头节举办地点? 丹霞山性博物馆要门票吗?韶关丹霞山性博物馆门票优惠吗? 建德下渚湖 秋意浓满江南湖畔 天降长假哪里去 杭州乡野寻清闲(组图) 不去著名景点 武汉的小地方也可以很美 在细节处发现美 武汉那些小景点推荐 十一内蒙古赏秋地推荐 青岛住宿好去处 享受海光山色 2013庐山旅游攻略及注意事项 山西旅游注意事项 武汉周边秋天烧烤地推荐之咸宁刘家桥_森林公园旅游攻略_森林公园旅游景点 普吉岛在哪个国家?在哪里?_东南亚旅游攻略_东南亚旅游景点 澳大利亚旅游注意事项有哪些?_澳大利亚旅游攻略_澳大利亚旅游景点 武汉周边秋天烧烤地推荐之马鞍山森林公园_森林公园旅游攻略_森林公园旅游景点 MSL-1144UB Datasheet MSL-1144UB Datasheet MSL-1144UG Datasheet MSL-1144UG Datasheet MSL-1144UOL Datasheet MSL-1144UOL Datasheet 大溪地度假村 大溪地度假村 大溪地度假村 贵州省旅游景点 贵州省旅游景点 贵州省旅游景点 鼓浪屿 攻略 鼓浪屿 攻略 鼓浪屿 攻略