Java Spring 交易回復詳解,javaspring

來源:互聯網
上載者:User

Java Spring 交易回復詳解,javaspring

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()是會復原的    } } 

   感謝您的閱讀!謝謝!

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.