Spring并不直接管理事務(wù),而是提供了多種事務(wù)管理器辈双。他們將事務(wù)管理的職責(zé)委托給Hibernate或者JTA等持久化機(jī)制所提供的相關(guān)平臺框架的事務(wù)來實(shí)現(xiàn)。
Spring事務(wù)管理的核心接口是PlatformTransactionManager柜砾。
事務(wù)管理器接口通過getTransaction(TransactionDefinition definition)方法根據(jù)指定的傳播行為返回當(dāng)前活動的事務(wù)或創(chuàng)建一個新的事務(wù)湃望,這個方法里面的參數(shù)是TransactionDefinition類,這個類就定義了一些基本的事務(wù)屬性痰驱。
主要方法有:
int getIsolationLevel();// 返回事務(wù)的隔離級別
String getName();// 返回事務(wù)的名稱
int getPropagationBehavior();// 返回事務(wù)的傳播行為
int getTimeout(); // 返回事務(wù)必須在多少秒內(nèi)完成
boolean isReadOnly(); // 事務(wù)是否只讀证芭,事務(wù)管理器能夠根據(jù)這個返回值進(jìn)行優(yōu)化,確保事務(wù)是只讀的
隔離級別
Innodb中的事務(wù)隔離級別實(shí)現(xiàn)原理
隔離級別是指若干個并發(fā)的事務(wù)之間的隔離程度担映,與我們開發(fā)時候主要相關(guān)的場景包括:臟讀取废士、重復(fù)讀、幻讀蝇完。
我們可以看 org.springframework.transaction.annotation.Isolation 枚舉類中定義了五個表示隔離級別的值:
public enum Isolation {
DEFAULT(-1),
READ_UNCOMMITTED(1),
READ_COMMITTED(2),
REPEATABLE_READ(4),
SERIALIZABLE(8);
}
DEFAULT:這是默認(rèn)值官硝,表示使用底層數(shù)據(jù)庫的默認(rèn)隔離級別。對大部分?jǐn)?shù)據(jù)庫而言短蜕,通常這值就是: READ_COMMITTED 氢架。
READ_UNCOMMITTED :該隔離級別表示一個事務(wù)可以讀取另一個事務(wù)修改但還沒有提交的數(shù)據(jù)。該級別不能防止臟讀和不可重復(fù)讀朋魔,因此很少使用該隔離級別岖研。
READ_COMMITTED :該隔離級別表示一個事務(wù)只能讀取另一個事務(wù)已經(jīng)提交的數(shù)據(jù)。該級別可以防止臟讀警检,這也是大多數(shù)情況下的推薦值孙援。
REPEATABLE_READ :該隔離級別表示一個事務(wù)在整個過程中可以多次重復(fù)執(zhí)行某個查詢,并且每次返回的記錄都相同解滓。即使在多次查詢之間有新增的數(shù)據(jù)滿足該查詢赃磨,這些新增的記錄也會被忽略。該級別可以防止臟讀和不可重復(fù)讀洼裤。
SERIALIZABLE :所有的事務(wù)依次逐個執(zhí)行邻辉,這樣事務(wù)之間就完全不可能產(chǎn)生干擾,也就是說腮鞍,該級別可以防止臟讀值骇、不可重復(fù)讀以及幻讀。但是這將嚴(yán)重影響程序的性能移国。通常情況下也不會用到該級別吱瘩。
指定方法:通過使用 isolation 屬性設(shè)置,例如:
@Transactional(isolation = Isolation.DEFAULT)
傳播行為
所謂事務(wù)的傳播行為是指迹缀,如果在開始當(dāng)前事務(wù)之前使碾,一個事務(wù)上下文已經(jīng)存在蜜徽,此時有若干選項(xiàng)可以指定一個事務(wù)性方法的執(zhí)行行為。
我們可以看 org.springframework.transaction.annotation.Propagation 枚舉類中定義了6個表示傳播行為的枚舉值:
public enum Propagation {
REQUIRED(0),
SUPPORTS(1),
MANDATORY(2),
REQUIRES_NEW(3),
NOT_SUPPORTED(4),
NEVER(5),
NESTED(6);
}
REQUIRED :如果當(dāng)前存在事務(wù)票摇,則加入該事務(wù)拘鞋;如果當(dāng)前沒有事務(wù),則創(chuàng)建一個新的事務(wù)矢门。
SUPPORTS :如果當(dāng)前存在事務(wù)盆色,則加入該事務(wù);如果當(dāng)前沒有事務(wù)祟剔,則以非事務(wù)的方式繼續(xù)運(yùn)行隔躲。
MANDATORY :如果當(dāng)前存在事務(wù),則加入該事務(wù)物延;如果當(dāng)前沒有事務(wù)宣旱,則拋出異常。
REQUIRES_NEW :創(chuàng)建一個新的事務(wù)叛薯,如果當(dāng)前存在事務(wù)响鹃,則把當(dāng)前事務(wù)掛起。
NOT_SUPPORTED :以非事務(wù)方式運(yùn)行案训,如果當(dāng)前存在事務(wù),則把當(dāng)前事務(wù)掛起粪糙。
NEVER :以非事務(wù)方式運(yùn)行强霎,如果當(dāng)前存在事務(wù),則拋出異常蓉冈。
NESTED :如果當(dāng)前存在事務(wù)城舞,則創(chuàng)建一個事務(wù)作為當(dāng)前事務(wù)的嵌套事務(wù)來運(yùn)行;如果當(dāng)前沒有事務(wù)寞酿,則該取值等價于 REQUIRED 家夺。
指定方法:通過使用 propagation 屬性設(shè)置,例如:
@Transactional(propagation = Propagation.REQUIRED)