1. 程式人生 > >spring 事務回滾

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

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

相關推薦

spring事務的多種方式

轉:https://www.cnblogs.com/zeng1994/p/8257763.html start 看下下面的說明,會對理解本人貼出的程式碼有幫助。 1.程式碼中事務控制的3種方式 程式設計式事務:就是直接在程式碼裡手動開啟事務,手動提交,手動回滾。優點就是可以靈活控制,缺點

spring 事務、程式碼

1、遇到的問題   當我們一個方法裡面有多個數據庫儲存操作的時候,中間的資料庫操作發生的錯誤。虛擬碼如下: public method() { Dao1.save(Person1); Dao1.save(Person2); Dao1.save(P

關於Spring事務@Transactional使用記錄

測試@Transactional的在處理異常時是否生效, 首先這裡隨便寫一個update的方法,對資料庫進行更新操作,然後在操作完之後拋一個異常 @Override @Transactional(rollbackFor = {Exception.class})

try catch異常丟擲與spring事務策略相關

將異常捕獲,並且在catch塊中不對事務做顯式提交(或其他應該做的操作如關閉資源等)=生吞掉異常; spring的事務邊界是在呼叫業務方法之前開始的,業務方法執行完畢之後來執行commit or rollback(Spring預設取決於是否丟擲runtime異常). 

Spring事務與異常

Spring的事務管理預設是針對unchecked exception回滾,也就是預設對Error異常和RuntimeException異常以及其子類進行事務回滾,且必須對丟擲異常,若使用try-catch對其異常捕獲則不會進行回滾,會把事物吞併!需要手動丟擲執行時異常才會回滾(Error

spring 事務

1、遇到的問題   當我們一個方法裡面有多個數據庫儲存操作的時候,中間的資料庫操作發生的錯誤。虛擬碼如下: public method() { Dao1.save(Person1); Dao1.save(Person2); Dao1.save(Person2);//假如這句發生了錯

hibernate的executeWithNativeSession(HibernateCallbk())、execute(new HibernateCallback())無法對spring事務

hibernate作為DAO使用如下的方法操作資料庫時,無法對spring在service層新增的事務進行回滾。 hibernateTemplate.executeWithNativeSession(new HibernateCallback(

異常型別 && spring事務

使用spring難免要用到spring的事務管理,要用事務管理又會很自然的選擇宣告式的事務管理,在spring的文件中說道,spring宣告式事務管理預設對非檢查型異常和執行時異常進行事務回滾,而對檢查型異常則不進行回滾操作。那麼什麼是檢查型異常什麼又是非檢查型異常呢?最簡單的判斷點有兩個:1.繼承自runt

java spring事務

spring 事務回滾 1、遇到的問題   當我們一個方法裡面有多個數據庫儲存操作的時候,中間的資料庫操作發生的錯誤。虛擬碼如下: public method() { Dao1.save(Person1); Dao1.save(Person2); Dao1.

java之Spring事務和Ehcache配置

弄了一大早,終於配好了事務,事務的掃描包配好,Ehcache就是切面的問題,一切問題也迎刃而解。。。 一、Spring事務回滾 1、applicationContext.xml中配置 <!--spring 掃包 @Service .....

Spring事務和異常類

1、異常的一些基本知識 異常的架構   異常的繼承結構:Throwable為基類,Error和Exception繼承Throwable。Error和RuntimeException及其子類成為未檢查異常(unchecked),其它異常成為已檢查異常(checked)。  

try{} catch(Exception e){}異常丟擲與Spring事務策略

Spring的事務管理預設只對出現執行期異常(java.lang.RuntimeException及其子類)進行回滾。 如果一個方法丟擲Exception或者Checked異常,Spring事務管理預設不進行回滾。 Throwable分為Error和Exc

Spring 實現部分事務

light back true prop 回滾 sage .class lba aaa 例如有業務需求,在catch異常後,catch塊內把異常的信息存入到數據庫,而catch外的數據全部回滾 try { ....... aaaService.save();

淺談Spring中的事務

spec style try 常見 產生原因 turn prop ret run 使用Spring管理事務過程中,碰到過一些坑,因此也稍微總結一下,方便後續查閱。1.代碼中事務控制的3種方式編程式事務:就是直接在代碼裏手動開啟事務,手動提交,手動回滾。優點就是可以靈

哪些異常是RuntimeException?Sql異常屬於RuntimeException嗎?Spring下SQL異常事務

tail 好的 duplicate 代碼 blog 後拋 ase owa 接口文檔 一,為什麽框架中根本沒有對Exception的一般子類進行回滾配置,異常發生時,事務都進行了回滾 ,說好的只會對RuntimeException(Unchecked 非受檢異常)回滾呢?

spring事務管理,基於xml配置完成事務spring中資料庫表中欄位名和pojo中屬性名不一致時候,實現RowMapper介面手動封裝

宣告使用JDK8,spring5.0.7, 測試說明: service 層 宣告介面進行轉賬,從A轉賬B ,然後對AB 進行更新操作,在事務中對find方法開啟 只讀許可權,無法進行更新操作,造成事務回滾進行測試事務; 主要測試方法:* void tra

spring手動事務

@Transactional(rollbackFor = { Exception.class }) public String methods() { String str = "-----------"; try { doSomethin

spring手動控制事務

在catch語塊中增加TransactionAspectSupport.currentTransactionStatus().setRollbackOnly(); import org.springframework.transaction.interceptor.TransactionAs

Spring @Transactional 事務機制

Srping 事務     在Spring 的世界裡面我們一般使用@Transactional 註解在對應方法上面宣告為一個事務方法。     但是在預設不寫@Transactional(rollbackFor = Exception.class)預設回滾Run

程式碼丟擲異常後進行事務的兩種方式(Spring @Transactional註解)

需求 在service層的某個方法中,在執行完一個對資料庫的寫方法後,丟擲異常,再執行另一個對資料庫的寫方法,虛擬碼如下: @Transactional public void func() { dao.write(pojo1); throw new Exception("異常"