解決方法A調(diào)用方法B的事務(wù)控制問(wèn)題
關(guān)于方法A調(diào)用方法B的事務(wù)控制問(wèn)題
實(shí)測(cè),不管是上圖還是下圖,得到的結(jié)論都是一致的,那就是:
前提是同一個(gè)類里的方法調(diào)用,controller層類調(diào)用service的B方法,事務(wù)自然會(huì)生效的
a.異常發(fā)生后,被調(diào)用方法(B)是否添加事務(wù)控制,對(duì)于事務(wù)的回滾是否并不產(chǎn)生影響
b.調(diào)用B的方法,添加了事務(wù)控制才能實(shí)現(xiàn)異常事務(wù)回滾,不管被調(diào)用方法B是否有事務(wù)控制
c.不管嵌套調(diào)用了多少個(gè)方法,只要最頂層方法(沒(méi)被同一個(gè)類中其他方法調(diào)用的且調(diào)用了同一個(gè)類里其他方法的方法),有事務(wù)控制,那么不管是哪個(gè)被調(diào)用的方法異常,整個(gè)調(diào)用的業(yè)務(wù)數(shù)據(jù)都回滾,不管被調(diào)用的方法是否有添加了事務(wù)的控制
其實(shí),最好的辦法是根據(jù)自己實(shí)際調(diào)用的情況,模擬測(cè)試一下就知道了。
同一個(gè)類的不同方法,A方法沒(méi)有@Transactional,B方法有@Transactional,A調(diào)用B方法,事務(wù)不起作用
問(wèn)題
同一個(gè)類的不同方法,A方法沒(méi)有@Transactional,B方法有@Transactional,A調(diào)用B方法,事務(wù)不起作用
原理解析
spring 在掃描bean的時(shí)候會(huì)掃描方法上是否包含@Transactional注解,如果包含,spring會(huì)為這個(gè)bean動(dòng)態(tài)地生成一個(gè)子類(即代理類,proxy),代理類是繼承原來(lái)那個(gè)bean的。
此時(shí),當(dāng)這個(gè)有注解的方法被調(diào)用的時(shí)候,實(shí)際上是由代理類來(lái)調(diào)用的,代理類在調(diào)用之前就會(huì)啟動(dòng)transaction。
然而,如果這個(gè)有注解的方法是被同一個(gè)類中的其他方法調(diào)用的,那么該方法的調(diào)用并沒(méi)有通過(guò)代理類,而是直接通過(guò)原來(lái)的那個(gè)bean,所以就不會(huì)啟動(dòng)transaction,我們看到的現(xiàn)象就是@Transactional注解無(wú)效。
? ? //接口 ? ? interface Service { ? ? ? ? void A(); ? ? ? ? void B(); ? ? } ? ? //目標(biāo)類,實(shí)現(xiàn)接口 ? ? class ServiceImpl implements Service { ? ? ? ? //no annotation here ? ? ? ? @Override ? ? ? ? public void A() { ? ? ? ? ? ? this.B(); ? ? ? ? } ?? ??? ?@Transactional ? ? ? ? @Override ? ? ? ? public void B() { ? ? ? ? ? ? System.out.println("execute doNeedTx in ServiceImpl"); ? ? ? ? } ? ? } ? ? //代理類,也要實(shí)現(xiàn)相同的接口 ? ? class ProxyByJdkDynamic implements Service { ? ? ? ? //包含目標(biāo)對(duì)象 ? ? ? ? private Service target; ? ? ? ? public ProxyByJdkDynamic(Service target) { ? ? ? ? ? ? this.target = target; ? ? ? ? } ? ? ? ? //目標(biāo)類中此方法帶注解,進(jìn)行特殊處理 ? ? ? ? @Override ? ? ? ? public void B() { ? ? ? ? ? ? //開啟事務(wù) ? ? ? ? ? ? System.out.println("-> create Tx here in Proxy"); ? ? ? ? ? ? //調(diào)用目標(biāo)對(duì)象的方法,該方法已在事務(wù)中了 ? ? ? ? ? ? target.B(); ? ? ? ? ? ? //提交事務(wù) ? ? ? ? ? ? System.out.println("<- commit Tx here in Proxy"); ? ? ? ? } ? ? ? ? //目標(biāo)類中此方法沒(méi)有注解,只做簡(jiǎn)單的調(diào)用 ? ? ? ? @Override ? ? ? ? public void A() { ? ? ? ? ? ? //直接調(diào)用目標(biāo)對(duì)象方法 ? ? ? ? ? ? target.A(); ? ? ? ? } ? ? }
那回到一開始的問(wèn)題,我們調(diào)用的方法A不帶注解,因此代理類不開事務(wù),而是直接調(diào)用目標(biāo)對(duì)象的方法。
當(dāng)進(jìn)入目標(biāo)對(duì)象的方法后,執(zhí)行的上下文已經(jīng)變成目標(biāo)對(duì)象本身了,因?yàn)槟繕?biāo)對(duì)象的代碼是我們自己寫的,和事務(wù)沒(méi)有半毛錢關(guān)系,此時(shí)你再調(diào)用帶注解的方法,照樣沒(méi)有事務(wù),只是一個(gè)普通的方法調(diào)用而已。
簡(jiǎn)單來(lái)說(shuō),內(nèi)部調(diào)用本類方法,不會(huì)再走代理了,所以B的事務(wù)不起作用
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
SpringMVC中Controller類數(shù)據(jù)響應(yīng)的方法
這篇文章主要介紹了SpringMVC中的數(shù)據(jù)響應(yīng)的問(wèn)題,主要來(lái)了解 Controller 類如何進(jìn)行數(shù)據(jù)響應(yīng)的,本文給大家介紹的非常詳細(xì),需要的朋友可以參考下2021-07-07Java利用TreeUtils工具類實(shí)現(xiàn)列表轉(zhuǎn)樹
在開發(fā)過(guò)程中,總有列表轉(zhuǎn)樹的需求,幾乎是項(xiàng)目的標(biāo)配,有沒(méi)有一種通用且跨項(xiàng)目的解決方式呢?本文將基于Java8的Lambda?表達(dá)式和Stream等知識(shí),使用TreeUtils工具類實(shí)現(xiàn)一行代碼完成列表轉(zhuǎn)樹這一通用型需求,需要的可以參考一下2022-11-11Spring Web MVC框架學(xué)習(xí)之配置Spring Web MVC
這一篇文章講的是Spring Web MVC各部分的配置方法,包括Java代碼配置和XML文件配置以及MVC命名空間的使用方法。2017-03-03Java老手該當(dāng)心的13個(gè)錯(cuò)誤
這篇文章主要介紹了Java老手該當(dāng)心的13個(gè)錯(cuò)誤,需要的朋友可以參考下2015-04-04SpringBoot項(xiàng)目多數(shù)據(jù)源及mybatis 駝峰失效的問(wèn)題解決方法
這篇文章主要介紹了SpringBoot項(xiàng)目多數(shù)據(jù)源及mybatis 駝峰失效的問(wèn)題解決方法,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-07-07淺談MyBatisPlus中LocalDateTime引發(fā)的一些問(wèn)題和解決辦法
MyBatisPlus進(jìn)行數(shù)據(jù)庫(kù)操作時(shí),我們經(jīng)常會(huì)遇到處理日期時(shí)間類型的需求,本文主要介紹了淺談MyBatisPlus中LocalDateTime引發(fā)的一些問(wèn)題和解決辦法,具有一定的參考價(jià)值,感興趣的可以了解一下2024-07-07Java?LocalDateTime獲取時(shí)間信息、格式化、轉(zhuǎn)換為數(shù)字時(shí)間戳代碼示例
其實(shí)我們?cè)贘ava項(xiàng)目中對(duì)日期進(jìn)行格式化,主要是利用一些日期格式化類,下面這篇文章主要給大家介紹了關(guān)于Java?LocalDateTime獲取時(shí)間信息、格式化、轉(zhuǎn)換為數(shù)字時(shí)間戳的相關(guān)資料,需要的朋友可以參考下2023-11-11關(guān)于Elasticsearch封裝公共索引增刪改查
索引是Elasticsearch中存儲(chǔ)數(shù)據(jù)的邏輯單元,類似于關(guān)系數(shù)據(jù)庫(kù)中的表,它包含多個(gè)文檔,每個(gè)文檔都是一個(gè)結(jié)構(gòu)化的JSON數(shù)據(jù)格式,在實(shí)際應(yīng)用中,索引的使用與配置可以依據(jù)不同的方案進(jìn)行,例如在Spring Boot項(xiàng)目中,可以選擇自動(dòng)配置或者手動(dòng)編寫配置類2024-10-10MybatisPlus字段自動(dòng)填充失效,填充值為null的解決方案
這篇文章主要介紹了MybatisPlus字段自動(dòng)填充失效,填充值為null的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-01-01