通過@Transactional進(jìn)行事務(wù)控制是我們?cè)谌粘i_發(fā)中非常常見的使用方式情妖,雖然用起來感覺很簡(jiǎn)單,但是其內(nèi)部的實(shí)現(xiàn)細(xì)節(jié)較多霹俺,稍微不注意就會(huì)出現(xiàn)莫名其妙的事務(wù)失效他挎。這里記錄一下其失效的六種場(chǎng)景。
最好不要加到接口上
雖然@Transactional可以添加到接口上界弧,但是如果此時(shí)將動(dòng)態(tài)代理由JDK改為CGLib的話凡蜻,會(huì)出現(xiàn)失效。因?yàn)镃GLib是直接對(duì)類進(jìn)行代理夹纫,導(dǎo)致接口不生效咽瓷。
1)數(shù)據(jù)庫(kù)引擎不支持,事務(wù)失效
MySQL數(shù)據(jù)庫(kù)默認(rèn)引擎為INNODB舰讹,此引擎支持事務(wù)茅姜。但是當(dāng)將引擎切換為MYISAM時(shí),由于該引擎不支持事務(wù),所以會(huì)出現(xiàn)失誤失效钻洒。
2)不能加載非public方法上
Spring AOP代理時(shí)奋姿,在其內(nèi)部最終會(huì)檢查目標(biāo)方法的修飾符是否為public,如果不是public則不會(huì)獲取@Transactional的屬性信息素标,導(dǎo)致事務(wù)失效称诗。但是在protected和private方法上雖然事務(wù)失效了,但不會(huì)出現(xiàn)任何報(bào)錯(cuò)头遭,此處需注意
protected TransactionAttribute computeTransactionAttribute(Method method,
Class<?> targetClass) {
// Don't allow no-public methods as required.
if (allowPublicMethodsOnly() && !Modifier.isPublic(method.getModifiers())) {
return null;
}
3)@Transactional中propagation屬性配置錯(cuò)誤
事務(wù)的傳播機(jī)制總共有七種寓免,分別為:
那么當(dāng)配置以下三種事務(wù)傳播機(jī)制時(shí),會(huì)導(dǎo)致事務(wù)失效
SUPPORTS:如果當(dāng)前有事務(wù)计维,則沿用當(dāng)前事務(wù)袜香,如果沒有,以非事務(wù)方式運(yùn)行
NOT_SUPPORTED:以非事務(wù)方式運(yùn)行鲫惶,如果當(dāng)前存在事務(wù)蜈首,則把當(dāng)前事務(wù)掛起
NEVER:以非事務(wù)方式運(yùn)行,如果當(dāng)前存在事務(wù)欠母,則拋出異常
4)使用自定義異常
Spring默認(rèn)拋出繼承自 RuntimeException 的異郴恫撸或者Error時(shí),才會(huì)出現(xiàn)事務(wù)回滾赏淌,其他類型的異常不會(huì)觸發(fā)事務(wù)回滾踩寇。如果需要拋出指定類型的異常,如自定義異常六水。同時(shí)需要事務(wù)回滾姑荷。需要通過rollbackFor 進(jìn)行指定。同時(shí)對(duì)于自定義異常的子類同樣也會(huì)觸發(fā)事務(wù)回滾缩擂。
@Transactional(rollbackFor= MyException.class)
5)同類中方法調(diào)用,事務(wù)失效
A方法沒有添加事務(wù)注解添寺,B方法添加事務(wù)注解胯盯。并且A調(diào)用B。此時(shí)當(dāng)外部調(diào)用A方法時(shí)计露,B方式事務(wù)不會(huì)生效博脑。因?yàn)樵赟pring AOP中,事務(wù)方法只有被當(dāng)前類以外的代碼調(diào)用票罐,Spring才會(huì)生成代理對(duì)象叉趣。
6)catch捕獲異常,事務(wù)失效
假設(shè)B中insertInfo執(zhí)行時(shí)會(huì)出現(xiàn)異常该押,但是A對(duì)下游的B出現(xiàn)的異常進(jìn)行的手動(dòng)捕獲并進(jìn)行處理疗杉,所以A會(huì)認(rèn)為當(dāng)前事務(wù)正常提交。此時(shí)就會(huì)出現(xiàn)B事務(wù)回滾蚕礼,但A執(zhí)行成功烟具。導(dǎo)致數(shù)據(jù)出現(xiàn)不一致梢什。