Spring的事務管理
事務原本是數據庫中的概念,在Dao層。但一般情況下,需要將事務提升到業務層,即Service層。這樣做是為了能夠使用事務的特性來管理具體的業務。
在Spring中通常可以通過以下兩種方式來實現對事務的管理:
● 使用Spring的事務注解管理事務
● 使用AspectJ的AOP配置管理事務
Spring的事務管理,主要用到兩個事務相關的接口。
事務管理器是PlatformTransactionManager接口對象。其主要用于完成事務的提交、回滾,及獲取事務的狀態信息。
A、 常用的兩個實現類
PlatformTransactionManager接口有兩個常用的實現類:
● DataSourceTransactionManager:使用JDBC或MyBatis進行數據庫操作時使用。
● HibernateTransactionManager:使用Hibernate進行持久化數據時使用。
B、 Spring的回滾方式(理解)
Spring事務的默認回滾方式是:發生運行時異常和error時回滾,發生受查(編譯)異常時提交。不過,對于受查異常,程序員也可以手工設置其回滾方式。
C、 回顧錯誤與異常(理解)
Throwable類是Java語言中所有錯誤或異常的超類。只有當對象是此類(或其子類之一)的實例時,才能通過Java虛擬機或者Java的throw語句拋出。
Error是程序在運行過程中出現的無法處理的錯誤,比如OutOfMemoryError、ThreadDeath、NoSuchMethodError等。當這些錯誤發生時,程序是無法處理(捕獲或拋出)的,JVM一般會終止線程。
程序在編譯和運行時出現的另一類錯誤稱之為異常,它是JVM通知程序員的一種方式。通過這種方式,讓程序員知道已經或可能出現錯誤,要求程序員對其進行處理。
異常分為運行時異常與受查異常。
運行時異常,是RuntimeException類或其子類,即只有在運行時才出現的異常。如,NullPointerException、ArrayIndexOutOfBoundsException、IllegalArgumentException 等均屬于運行時異常。這些異常由JVM拋出,在編譯時不要求必須處理(捕獲或拋出)。但,只要代碼編寫足夠仔細,程序足夠健壯,運行時異常是可以避免的。
受查異常,也叫編譯時異常,即在代碼編寫時要求必須捕獲或拋出的異常,若不處理,則無法通過編譯。如SQLException,ClassNotFoundException,IOException等都屬于受查異常。
RuntimeException及其子類以外的異常,均屬于受查異常。當然,用戶自定義的Exception的子類,即用戶自定義的異常也屬受查異常。程序員在定義異常時,只要未明確聲明定義的為RuntimeException的子類,那么定義的就是受查異常。
事務定義接口TransactionDefinition中定義了事務描述相關的三類常量:事務隔離級別、事務傳播行為、事務默認超時時限,及對它們的操作。
A、 定義了五個事務隔離級別常量(掌握)
這些常量均是以ISOLATION_開頭。即形如ISOLATION_XXX。
DEFAULT:采用DB默認的事務隔離級別。MySql的默認為REPEATABLE_READ; Oracle默認為READ_COMMITTED。
READ_UNCOMMITTED:讀未提交。未解決任何并發問題。
READ_COMMITTED:讀已提交。解決臟讀,存在不可重復讀與幻讀。
REPEATABLE_READ:可重復讀。解決臟讀、不可重復讀,存在幻讀
SERIALIZABLE:串行化。不存在并發問題。
B、 定義了七個事務傳播行為常量(掌握)
所謂事務傳播行為是指,處于不同事務中的方法在相互調用時,執行期間事務的維護情況。如,A事務中的方法doSome()調用B事務中的方法doOther(),在調用執行期間事務的維護情況,就稱為事務傳播行為。事務傳播行為是加在方法上的。
事務傳播行為常量都是以PROPAGATION_ 開頭,形如PROPAGATION_XXX。
PROPAGATION_REQUIRED
PROPAGATION_REQUIRES_NEW
PROPAGATION_SUPPORTS
PROPAGATION_MANDATORY
PROPAGATION_NESTED
PROPAGATION_NEVER
PROPAGATION_NOT_SUPPORTED
a、 PROPAGATION_REQUIRED:
指定的方法必須在事務內執行。若當前存在事務,就加入到當前事務中;若當前沒有事務,則創建一個新事務。這種傳播行為是最常見的選擇,也是Spring默認的事務傳播行為。
如該傳播行為加在doOther()方法上。若doSome()方法在調用doOther()方法時就是在事務內運行的,則doOther()方法的執行也加入到該事務內執行。若doSome()方法在調用doOther()方法時沒有在事務內執行,則doOther()方法會創建一個事務,并在其中執行。
b、 PROPAGATION_SUPPORTS
指定的方法支持當前事務,但若當前沒有事務,也可以以非事務方式執行。
c、 PROPAGATION_REQUIRES_NEW
總是新建一個事務,若當前存在事務,就將當前事務掛起,直到新事務執行完畢。
C、定義了默認事務超時時限
常量TIMEOUT_DEFAULT定義了事務底層默認的超時時限,sql語句的執行時長。
注意,事務的超時時限起作用的條件比較多,且超時的時間計算點較復雜。所以,該值一般就使用默認值即可。