spring 事務回滾

一、遇到的問題spring

  當咱們一個方法裏面有多個數據庫保存操做的時候,中間的數據庫操做發生的錯誤。僞代碼以下:數據庫

public method() {
    Dao1.save(Person1);
    Dao1.save(Person2);

    Dao1.save(Person2);//假如這句發生了錯誤,前面的兩個對象會被保存到數據庫中
    Dao1.save(Person2);
}

  期待的狀況:發生錯誤以前的全部數據庫保存操做都回滾,即不保存編程

  正常狀況:前面的數據庫操做會被執行,而發生數據庫操做錯誤開始及以後的全部的數據保存操做都將失敗。這樣子應該都不是咱們要的結果吧。架構

  當遇到這種狀況,咱們就可使用Spring的事務解決這個問題。ui

二、異常的一些基本知識spa

1) 異常的架構code

  異常的繼承結構:Throwable爲基類,Error和Exception繼承Throwable,RuntimeException和IOException等繼承Exception。Error和RuntimeException及其子類成爲未檢查異常(unchecked),其它異常成爲已檢查異常(checked)。 orm

2)Error異常對象

  Error表示程序在運行期間出現了十分嚴重、不可恢復的錯誤,在這種狀況下應用程序只能停止運行,例如JAVA 虛擬機出現錯誤。Error是一種unchecked Exception,編譯器不會檢查Error是否被處理,在程序中不用捕獲Error類型的異常。通常狀況下,在程序中也不該該拋出Error類型的異常。blog

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進行處理,否則不能經過編譯。

三、實例

  這裏使用的事務配置以下:

 <!-- 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()是會回滾的  
     }  
}  

   致謝:感謝您的閱讀!轉文請加原文連接,謝謝!

相關文章
相關標籤/搜索