|
本文討論下Spring注解@Transactional 及其隔離(isolation)和傳播(propagation)屬性的設(shè)置. 1. @Transactional注解@Transactional注解可以用在數(shù)據(jù)庫(kù)事務(wù)操作的方法上。并可以設(shè)置事務(wù)的相關(guān)屬性:隔離屬性(isolation), 超時(shí)屬性(timeout), 只讀屬性(read-only)以及回滾條件,也可以指定事務(wù)管理器。 1.1. 實(shí)現(xiàn)細(xì)節(jié)Spring創(chuàng)建代理或操縱類字節(jié)碼來(lái)管理事務(wù)的創(chuàng)建、提交和回滾。使用代理方式Spring會(huì)忽略內(nèi)部方法調(diào)用事務(wù),即使有@Transactional注解也會(huì)被忽略,代理需要通過(guò)其他類進(jìn)行調(diào)用。 例如,一個(gè)方法callMethod并標(biāo)記了 @Transactional注解,Spring會(huì)包裝一些事務(wù)管理代碼環(huán)繞在方法執(zhí)行過(guò)程: createTransactionIfNecessary();
try {
callMethod();
commitTransactionAfterReturning();
} catch (exception) {
completeTransactionAfterThrowing();
throw exception;
}
1.2. 使用@Transactional注解事務(wù)注解可以在接口、類或直接在方法上。實(shí)際會(huì)按照優(yōu)先級(jí)進(jìn)行覆蓋,從低到高優(yōu)先級(jí)為:接口、父類、類、接口方法、父類方法、類方法。 對(duì)于類級(jí)別注解,則Spring應(yīng)用注解設(shè)置至所有沒(méi)有標(biāo)記注解的public方法,如果在private、protecte方法上使用注解,Spring會(huì)忽略。 下面通過(guò)示例說(shuō)明: @Transactional
public interface TransferService {
void transfer(String user1, String user2, double val);
}
通常不建議在接口上設(shè)置事務(wù),但在Spring Data @Repository情況下是可行的。這里在類上增加注解覆蓋接口或父類的設(shè)置: @Service
@Transactional
public class TransferServiceImpl implements TransferService {
@Override
public void transfer(String user1, String user2, double val) {
// ...
}
}
如果在方法增加注解則覆蓋類定義: @Transactional
public void transfer(String user1, String user2, double val) {
// ...
}
2. 事務(wù)傳播傳播性定義業(yè)務(wù)邏輯的事務(wù)邊界。Spring根據(jù)傳播性設(shè)置負(fù)責(zé)啟動(dòng)或暫停事務(wù)。 Spring根據(jù)傳播屬性調(diào)用TransactionManager::getTransaction 方法獲取或創(chuàng)建事務(wù)。它支持所有類型的TransactionManager的部分傳播屬性,一些傳播屬性僅被TransactionManager特定實(shí)現(xiàn)支持。下面詳細(xì)描述不同傳播屬性。 2.1. REQUIREDREQUIRED是缺省屬性。Spring檢查是否有活動(dòng)事務(wù),如果沒(méi)有則創(chuàng)建新的事務(wù),否則業(yè)務(wù)邏輯追加至當(dāng)前活動(dòng)事務(wù)中: @Transactional(propagation = Propagation.REQUIRED)
public void requiredExample(String user) {
// ...
}
缺省屬性也可以不指定: @Transactional
public void requiredExample(String user) {
// ...
}
對(duì)應(yīng)偽代碼如下: if (isExistingTransaction()) {
if (isValidateExistingTransaction()) {
validateExisitingAndThrowExceptionIfNotValid();
}
return existing;
}
return createNewTransaction();
2.2. SUPPORTSSUPPORTS屬性,Spring首先檢查是否有活動(dòng)事務(wù)存在,存在則使用,反之,則無(wú)事務(wù)進(jìn)行執(zhí)行。 @Transactional(propagation = Propagation.SUPPORTS)
public void supportsExample(String user) {
// ...
}
對(duì)應(yīng)偽代碼: if (isExistingTransaction()) {
if (isValidateExistingTransaction()) {
validateExisitingAndThrowExceptionIfNotValid();
}
return existing;
}
return emptyTransaction;
2.3. MANDATORY當(dāng)屬性設(shè)置為MANDATORY屬性時(shí),如果存在活動(dòng)事務(wù),則使用之。反之拋異常,即強(qiáng)制使用事務(wù)。 @Transactional(propagation = Propagation.MANDATORY)
public void mandatoryExample(String user) {
// ...
}
對(duì)應(yīng)偽代碼: if (isExistingTransaction()) {
if (isValidateExistingTransaction()) {
validateExisitingAndThrowExceptionIfNotValid();
}
return existing;
}
throw IllegalTransactionStateException;
2.4. NEVERNEVER屬性的邏輯:如果存在活動(dòng)事務(wù)則拋異常。 @Transactional(propagation = Propagation.NEVER)
public void neverExample(String user) {
// ...
}
對(duì)應(yīng)偽代碼: if (isExistingTransaction()) {
throw IllegalTransactionStateException;
}
return emptyTransaction;
2.5. NOT_SUPPORTED如果存在活動(dòng)事務(wù)則Spring首先掛起當(dāng)前事務(wù),然后業(yè)務(wù)邏輯在無(wú)事務(wù)下執(zhí)行。 @Transactional(propagation = Propagation.NOT_SUPPORTED)
public void notSupportedExample(String user) {
// ...
}
JTATransactionManager支持開箱即用的事務(wù)掛起。其他方法通過(guò)持有對(duì)事務(wù)引用,然后從線程上下文中清除它來(lái)模擬掛起。 2.6. REQUIRES_NEWREQUIRES_NEW屬性,如果存在活動(dòng)事務(wù)則Spring掛起當(dāng)前事務(wù),然后創(chuàng)建新的事務(wù)。 @Transactional(propagation = Propagation.REQUIRES_NEW)
public void requiresNewExample(String user) {
// ...
}
與NOT_SUPPORTED類似,我們需要JTATransactionManager來(lái)執(zhí)行實(shí)際的事務(wù)掛起。 偽代碼如下: if (isExistingTransaction()) {
suspend(existing);
try {
return createNewTransaction();
} catch (exception) {
resumeAfterBeginException();
throw exception;
}
}
return createNewTransaction();
2.7. NESTED對(duì)于NESTED屬性,Spring檢查是否存在事務(wù),如果存在則標(biāo)記保存點(diǎn)。意味著如果后續(xù)業(yè)務(wù)執(zhí)行遇到異常,那么回滾至保存點(diǎn)。如果沒(méi)有活動(dòng)事務(wù)時(shí)與REQUIRED屬性一樣。 DataSourceTransactionManager 支持該屬性,一些JTATransactionManager的實(shí)現(xiàn)可能也支持。JpaTransactionManager僅對(duì)JDBC連接支持NESTED屬性,如果設(shè)置nestedTransactionAllowed 屬性為true且JDBC驅(qū)動(dòng)支持保存點(diǎn),那么JPA事務(wù)中的JDBC代碼也工作。屬性設(shè)置如下: @Transactional(propagation = Propagation.NESTED)
public void nestedExample(String user) {
// ...
}
3. 事務(wù)隔離隔離屬性是ACID ( Atomicity, Consistency, Isolation及 Durability)其中之一,隔離描述并發(fā)事務(wù)應(yīng)用的更改如何對(duì)彼此可見。每種隔離級(jí)別防止事務(wù)中零個(gè)或多個(gè)并發(fā)副作用: 臟讀(Dirty read): 讀取并發(fā)事務(wù)中未提交的信息 不可重復(fù)讀(Nonrepeatable read): 如果并發(fā)事務(wù)更新相同行并提交,重復(fù)讀一行獲得不同值 幻讀(Phantom read): 如果其他事務(wù)增加或刪除查詢行并提交,則重復(fù)查詢一定范圍記錄返回值不同
我們可以通過(guò)@Transactional::isolation設(shè)置事務(wù)的隔離級(jí)別。Spring提供了5個(gè)枚舉值:DEFAULT, READ_UNCOMMITTED, READ_COMMITTED, REPEATABLE_READ, SERIALIZABLE。 3.1. 缺省隔離屬性當(dāng)Spring創(chuàng)建新事務(wù)時(shí),缺省隔離級(jí)別使用RDBMS,因此改變數(shù)據(jù)庫(kù)時(shí)應(yīng)該注意。 我們也應(yīng)該考慮使用不同隔離屬性調(diào)用一組方法的場(chǎng)景,正常流程隔離僅應(yīng)用于新事務(wù)創(chuàng)建時(shí)。因此出于某種原因,不想讓一個(gè)方法在不同的隔離狀態(tài)下執(zhí)行,我們必須將TransactionManager::setValidateExistingTransaction設(shè)置為true。偽代碼如下: if (isolationLevel != ISOLATION_DEFAULT) {
if (currentTransactionIsolationLevel() != isolationLevel) {
throw IllegalTransactionStateException
}
}
下面看看其他隔離級(jí)別。 3.2. READ_UNCOMMITTEDREAD_UNCOMMITTED是最低的隔離級(jí)別,最大化允許并發(fā)訪問(wèn)。 它受到上述三種并發(fā)性副作用的影響。具有此隔離的事務(wù)將讀取其他并發(fā)事務(wù)未提交數(shù)據(jù),此外不可重復(fù)讀取和幻讀都可能發(fā)生。因此我們可以在重新讀取行或重新執(zhí)行范圍查詢時(shí)獲得不同的結(jié)果??梢栽诜椒ɑ蝾惿显O(shè)置隔離級(jí)別: @Transactional(isolation = Isolation.READ_UNCOMMITTED)
public void log(String message) {
// ...
}
Postgres 不支持 READ_UNCOMMITTED 隔離屬性,會(huì)用 READ_COMMITED 代替。Oracle 也不支持 READ_UNCOMMITTED。 3.3. READ_COMMITTED第二級(jí)隔離READ_COMMITTED可以防止臟讀,其他并發(fā)副作用仍可能發(fā)生。并發(fā)事務(wù)的未提交改變沒(méi)有影響,但已提交的改變?cè)俅尾樵円矔?huì)改變。 設(shè)置隔離代碼: @Transactional(isolation = Isolation.READ_COMMITTED)
public void log(String message){
// ...
}
READ_COMMITTED Postgres, SQL Server及Oracle的缺省級(jí)別。 3.4. REPEATABLE_READ第三個(gè)隔離級(jí)別是REPEATABLE_READ,防止臟讀和不可重復(fù)讀,因此不會(huì)受并發(fā)事務(wù)未提交改變影響。當(dāng)重復(fù)查詢行不會(huì)得到不同結(jié)果,但可能獲得新增記錄或刪除部分行。 該級(jí)別可以防止丟失更新,當(dāng)兩個(gè)或多個(gè)并發(fā)事務(wù)讀并更新相同行會(huì)發(fā)生丟失更新。REPEATABLE_READ根本不允許同時(shí)訪問(wèn)行,因此丟失更新不會(huì)發(fā)生。 設(shè)置代碼: @Transactional(isolation = Isolation.REPEATABLE_READ)
public void log(String message){
// ...
}
REPEATABLE_READ Mysql的缺省級(jí)別,Oracle 不支持 REPEATABLE_READ。 3.5. SERIALIZABLESERIALIZABLE 是最高隔離級(jí)別??梢苑乐股鲜鏊械膯?wèn)題,但也導(dǎo)致最低并發(fā)訪問(wèn)效率,因?yàn)椴l(fā)事務(wù)按照順序執(zhí)行。也就是并發(fā)執(zhí)行一組SERIALIZABLE級(jí)別事務(wù)與順序執(zhí)行結(jié)果一樣。設(shè)置代碼: @Transactional(isolation = Isolation.SERIALIZABLE)
public void log(String message){
// ...
}
4. 總結(jié)本文我們探討了事務(wù)的傳遞與隔離屬性,并詳細(xì)解釋了不同屬性的含義及對(duì)并發(fā)事務(wù)的影響。
|