從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住了
代碼中手動catch了異常,然后又未拋出來,此時事務(wù)就不生效了。
二、方法不是 public 的
以下來自Spring官方文檔
大概意思就是 @Transactional 只能用于 public 的方法上,否則事務(wù)不會失效,如果要用在非 public 方法上,可以開啟 AspectJ 代理模式。
TransactionInterceptor#invoke
TransactionAspectSupport#invokeWithinTransaction
AbstractFallbackTransactionAttributeSource#getTransactionAttribute
AbstractFallbackTransactionAttributeSource#computeTransactionAttribute
allowPublicMethodsOnly方法由子類AnnotationTransactionAttributeSource實(shí)現(xiàn),該子類方法中默認(rèn)是true,所以當(dāng)你加了事務(wù)注解的方法不是public時,該方法直接返回null
以上只說明 在spring的事務(wù)管理中存在這樣一個 public的判斷,至于后邊為啥由于這個判斷導(dǎo)致沒有織入事務(wù),并沒有說明;
public導(dǎo)致失效原因
三、內(nèi)部方法間調(diào)用導(dǎo)致事務(wù)失效
因?yàn)閟pring聲明式事務(wù)是基于AOP實(shí)現(xiàn)的,是使用動態(tài)代理來達(dá)到事務(wù)管理的目的,當(dāng)前類調(diào)用的方法上面加@Transactional 這個是沒有任何作用的,因?yàn)檎{(diào)用這個方法的是this,沒有經(jīng)過 Spring 的代理類。
解決方案一:
再聲明一個service,自己注入自己,將內(nèi)部調(diào)用改為外部調(diào)用
解決方案二:
解決方案三:
使用編程式事務(wù)
四、異常類型是否配置正確
拋出RuntimeException異常,事務(wù)生效
拋出Exception異常,事務(wù)不生效
我們看下TransactionInterceptor類
TransactionInterceptor#invoke
TransactionAspectSupport#invokeWithinTransaction
TransactionAspectSupport#completeTransactionAfterThrowing
配置了這個,Exception異常的事務(wù),就會生效
五、異常被catch住
代碼中手動catch了異常,然后又未拋出來,此時事務(wù)就不生效了。
解決方法:要么不catch需要回滾的異常,要么catch之后再拋出,要么手動回滾
解決方案一:
catch之后往外拋異常
解決方案二:
catch之后,設(shè)置手動回滾
到此這篇關(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這兩個運(yùn)算符進(jìn)行,本文就來介紹一下如何實(shí)現(xiàn),感興趣的可以了解一下2023-09-09mybatis plus表的創(chuàng)建時間和修改時間的操作方法
這篇文章主要介紹了mybatis plus表的創(chuàng)建時間和修改時間的實(shí)現(xiàn)方法,本文給大家分享兩種方法,每種方法通過實(shí)例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下2021-09-09spring Boot查詢數(shù)據(jù)分頁顯示的方法實(shí)例
這篇文章主要給大家介紹了關(guān)于spring Boot查詢數(shù)據(jù)分頁顯示的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家學(xué)習(xí)或者使用spring Boot具有一定的參考學(xué)習(xí)價值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08Java實(shí)現(xiàn)鎖定某個變量的幾種方式示例詳解
這篇文章主要為大家介紹了Java實(shí)現(xiàn)鎖某個變量的幾種方式示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-09-09Spring Boot中的那些條件判斷的實(shí)現(xiàn)方法
這篇文章主要介紹了Spring Boot中的那些條件判斷的實(shí)現(xiàn)方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-04-04