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