從Spring源碼解析事務(wù)失效的原因
一、前言
1.Bean是否是代理對象
2.入口函數(shù)是否是public的
3.數(shù)據(jù)庫是否支持事務(wù)(Mysql的Mvlsam不支持事務(wù)),行鎖才支持事務(wù)
4.切點(diǎn)是否配置正確
5.內(nèi)部方法間調(diào)用導(dǎo)致事務(wù)失效
因?yàn)閠his不是代理對象,可以配置 expose-proxy="true",就可以通過AopContext.currentProxy()獲取到當(dāng)前類的代理對象。
<! -- expose-proxy="true”類內(nèi)部可以獲取到當(dāng)前類的代理對象--><aop: aspectj-autoproxy expose-proxy="true" /> @Enab1eAspect3AutoProxy(exposeProxy = true)
也可以注入當(dāng)前bean
6.異常類型是否配置正確
默認(rèn)只支持RuntimeException和Error,不支持檢查異常
想要支持檢查異常需要配置rollbackFor
@Transactional(rollbackFor = Exception.class)
7.異常被catch住了
代碼中手動(dòng)catch了異常,然后又未拋出來,此時(shí)事務(wù)就不生效了。
二、方法不是 public 的
以下來自Spring官方文檔

大概意思就是 @Transactional 只能用于 public 的方法上,否則事務(wù)不會(huì)失效,如果要用在非 public 方法上,可以開啟 AspectJ 代理模式。

TransactionInterceptor#invoke

TransactionAspectSupport#invokeWithinTransaction

AbstractFallbackTransactionAttributeSource#getTransactionAttribute

AbstractFallbackTransactionAttributeSource#computeTransactionAttribute

allowPublicMethodsOnly方法由子類AnnotationTransactionAttributeSource實(shí)現(xiàn),該子類方法中默認(rèn)是true,所以當(dāng)你加了事務(wù)注解的方法不是public時(shí),該方法直接返回null
以上只說明 在spring的事務(wù)管理中存在這樣一個(gè) public的判斷,至于后邊為啥由于這個(gè)判斷導(dǎo)致沒有織入事務(wù),并沒有說明;
public導(dǎo)致失效原因
三、內(nèi)部方法間調(diào)用導(dǎo)致事務(wù)失效

因?yàn)閟pring聲明式事務(wù)是基于AOP實(shí)現(xiàn)的,是使用動(dòng)態(tài)代理來達(dá)到事務(wù)管理的目的,當(dāng)前類調(diào)用的方法上面加@Transactional 這個(gè)是沒有任何作用的,因?yàn)檎{(diào)用這個(gè)方法的是this,沒有經(jīng)過 Spring 的代理類。
解決方案一:
再聲明一個(gè)service,自己注入自己,將內(nèi)部調(diào)用改為外部調(diào)用

解決方案二:







解決方案三:
使用編程式事務(wù)

四、異常類型是否配置正確
拋出RuntimeException異常,事務(wù)生效
拋出Exception異常,事務(wù)不生效




我們看下TransactionInterceptor類
TransactionInterceptor#invoke

TransactionAspectSupport#invokeWithinTransaction

TransactionAspectSupport#completeTransactionAfterThrowing



配置了這個(gè),Exception異常的事務(wù),就會(huì)生效

五、異常被catch住
代碼中手動(dòng)catch了異常,然后又未拋出來,此時(shí)事務(wù)就不生效了。
解決方法:要么不catch需要回滾的異常,要么catch之后再拋出,要么手動(dòng)回滾

解決方案一:
catch之后往外拋異常

解決方案二:
catch之后,設(shè)置手動(dòng)回滾

到此這篇關(guān)于從Spring源碼解析事務(wù)失效的原因的文章就介紹到這了,更多相關(guān)Spring事務(wù)失效內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
java實(shí)現(xiàn)字符串like和not?like的使用示例
在Java中,我們經(jīng)常需要對字符串進(jìn)行模式匹配操作,字符串的模式匹配通常使用like和not?like這兩個(gè)運(yùn)算符進(jìn)行,本文就來介紹一下如何實(shí)現(xiàn),感興趣的可以了解一下2023-09-09
mybatis plus表的創(chuàng)建時(shí)間和修改時(shí)間的操作方法
這篇文章主要介紹了mybatis plus表的創(chuàng)建時(shí)間和修改時(shí)間的實(shí)現(xiàn)方法,本文給大家分享兩種方法,每種方法通過實(shí)例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下2021-09-09
spring Boot查詢數(shù)據(jù)分頁顯示的方法實(shí)例
這篇文章主要給大家介紹了關(guān)于spring Boot查詢數(shù)據(jù)分頁顯示的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家學(xué)習(xí)或者使用spring Boot具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08
Java實(shí)現(xiàn)鎖定某個(gè)變量的幾種方式示例詳解
這篇文章主要為大家介紹了Java實(shí)現(xiàn)鎖某個(gè)變量的幾種方式示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-09-09
Spring Boot中的那些條件判斷的實(shí)現(xiàn)方法
這篇文章主要介紹了Spring Boot中的那些條件判斷的實(shí)現(xiàn)方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-04-04

