Spring事務(wù)一個(gè)被訛傳很廣說法是:一個(gè)事務(wù)方法不應(yīng)該調(diào)用另一個(gè)事務(wù)方法,否則將產(chǎn)生兩個(gè)事務(wù)秆剪。結(jié)果造成開發(fā)人員在設(shè)計(jì)事務(wù)方法時(shí)束手束腳赊淑,生怕一不小心就踩到地雷。?
其實(shí)這是不認(rèn)識(shí)Spring事務(wù)傳播機(jī)制而造成的誤解仅讽,Spring對(duì)事務(wù)控制的支持統(tǒng)一在TransactionDefinition類中描述陶缺,該類有以下幾個(gè)重要的接口方法:?
int getPropagationBehavior():事務(wù)的傳播行為
int getIsolationLevel():事務(wù)的隔離級(jí)別
int getTimeout():事務(wù)的過期時(shí)間
boolean isReadOnly():事務(wù)的讀寫特性
?? 很明顯,除了事務(wù)的傳播行為外洁灵,事務(wù)的其他特性Spring是借助底層資源的功能來完成的组哩,Spring無非只充當(dāng)個(gè)代理的角色。但是事務(wù)的傳播行為卻是Spring憑借自身的框架提供的功能处渣,是Spring提供給開發(fā)者最珍貴的禮物伶贰,訛傳的說法玷污了Spring事務(wù)框架最美麗的光環(huán)。?
?? 所謂事務(wù)傳播行為就是多個(gè)事務(wù)方法相互調(diào)用時(shí)罐栈,事務(wù)如何在這些方法間傳播黍衙。Spring支持以下7種事務(wù)傳播行為。?
PROPAGATION_REQUIRED:如果當(dāng)前沒有事務(wù)荠诬,就新建一個(gè)事務(wù)琅翻,如果已經(jīng)存在一個(gè)事務(wù)褂始,就加入到這個(gè)事務(wù)中杂伟。這是最常見的選擇。
PROPAGATION_SUPPORTS:支持當(dāng)前事務(wù)拯钻,如果當(dāng)前沒有事務(wù)钧嘶,就以非事務(wù)方式執(zhí)行棠众。
PROPAGATION_MANDATORY:使用當(dāng)前的事務(wù),如果當(dāng)前沒有事務(wù),就拋出異常闸拿。
PROPAGATION_REQUIRES_NEW:新建事務(wù)空盼,如果當(dāng)前存在事務(wù),把當(dāng)前事務(wù)掛起新荤。
PROPAGATION_NOT_SUPPORTED:以非事務(wù)方式執(zhí)行操作揽趾,如果當(dāng)前存在事務(wù),就把當(dāng)前事務(wù)掛起苛骨。
PROPAGATION_NEVER:以非事務(wù)方式執(zhí)行篱瞎,如果當(dāng)前存在事務(wù),則拋出異常痒芝。
PROPAGATION_NESTED:如果當(dāng)前存在事務(wù)俐筋,則在嵌套事務(wù)內(nèi)執(zhí)行。如果當(dāng)前沒有事務(wù)吼野,則執(zhí)行與PROPAGATION_REQUIRED類似的操作校哎。
?? Spring默認(rèn)的事務(wù)傳播行為是PROPAGATION_REQUIRED,它適合絕大多數(shù)的情況瞳步,如果多個(gè)ServiveX#methodX()均工作在事務(wù)環(huán)境下(即均被Spring事務(wù)增強(qiáng))闷哆,且程序中存在如下的調(diào)用鏈:Service1#method1()->Service2#method2()->Service3#method3(),那么這3個(gè)服務(wù)類的3個(gè)方法通過Spring的事務(wù)傳播機(jī)制都工作在同一個(gè)事務(wù)中单起。?
相互嵌套的服務(wù)方法?
?? 我們來看一下實(shí)例抱怔,UserService#logon()方法內(nèi)部調(diào)用了UserService#updateLastLogon Time()和ScoreService#addScore()方法,這兩個(gè)類都繼承于BaseService嘀倒。它們之間的類結(jié)構(gòu)如下圖所示:
? ?UserService#logon()方法內(nèi)部調(diào)用了ScoreService#addScore()的方法屈留,兩者都分別通過Spring AOP進(jìn)行了事務(wù)增強(qiáng),則它們工作于同一事務(wù)中测蘑。來看具體的代碼:?