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