首頁  >  文章  >  类库下载  >  spring 事務回滾

spring 事務回滾

高洛峰
高洛峰原創
2016-10-17 09:31:371987瀏覽

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

spring 事務回滾

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的異常都是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,可以看下面的源碼

spring 事務回滾

  程式中手動捕捉異常

@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()是会回滚的  
     }  
}


陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn