解決方法A調用方法B的事務控制問題
關于方法A調用方法B的事務控制問題
實測,不管是上圖還是下圖,得到的結論都是一致的,那就是:
前提是同一個類里的方法調用,controller層類調用service的B方法,事務自然會生效的
a.異常發(fā)生后,被調用方法(B)是否添加事務控制,對于事務的回滾是否并不產生影響
b.調用B的方法,添加了事務控制才能實現異常事務回滾,不管被調用方法B是否有事務控制
c.不管嵌套調用了多少個方法,只要最頂層方法(沒被同一個類中其他方法調用的且調用了同一個類里其他方法的方法),有事務控制,那么不管是哪個被調用的方法異常,整個調用的業(yè)務數據都回滾,不管被調用的方法是否有添加了事務的控制
其實,最好的辦法是根據自己實際調用的情況,模擬測試一下就知道了。
同一個類的不同方法,A方法沒有@Transactional,B方法有@Transactional,A調用B方法,事務不起作用
問題
同一個類的不同方法,A方法沒有@Transactional,B方法有@Transactional,A調用B方法,事務不起作用
原理解析
spring 在掃描bean的時候會掃描方法上是否包含@Transactional注解,如果包含,spring會為這個bean動態(tài)地生成一個子類(即代理類,proxy),代理類是繼承原來那個bean的。
此時,當這個有注解的方法被調用的時候,實際上是由代理類來調用的,代理類在調用之前就會啟動transaction。
然而,如果這個有注解的方法是被同一個類中的其他方法調用的,那么該方法的調用并沒有通過代理類,而是直接通過原來的那個bean,所以就不會啟動transaction,我們看到的現象就是@Transactional注解無效。
? ? //接口 ? ? interface Service { ? ? ? ? void A(); ? ? ? ? void B(); ? ? } ? ? //目標類,實現接口 ? ? 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"); ? ? ? ? } ? ? } ? ? //代理類,也要實現相同的接口 ? ? class ProxyByJdkDynamic implements Service { ? ? ? ? //包含目標對象 ? ? ? ? private Service target; ? ? ? ? public ProxyByJdkDynamic(Service target) { ? ? ? ? ? ? this.target = target; ? ? ? ? } ? ? ? ? //目標類中此方法帶注解,進行特殊處理 ? ? ? ? @Override ? ? ? ? public void B() { ? ? ? ? ? ? //開啟事務 ? ? ? ? ? ? System.out.println("-> create Tx here in Proxy"); ? ? ? ? ? ? //調用目標對象的方法,該方法已在事務中了 ? ? ? ? ? ? target.B(); ? ? ? ? ? ? //提交事務 ? ? ? ? ? ? System.out.println("<- commit Tx here in Proxy"); ? ? ? ? } ? ? ? ? //目標類中此方法沒有注解,只做簡單的調用 ? ? ? ? @Override ? ? ? ? public void A() { ? ? ? ? ? ? //直接調用目標對象方法 ? ? ? ? ? ? target.A(); ? ? ? ? } ? ? }
那回到一開始的問題,我們調用的方法A不帶注解,因此代理類不開事務,而是直接調用目標對象的方法。
當進入目標對象的方法后,執(zhí)行的上下文已經變成目標對象本身了,因為目標對象的代碼是我們自己寫的,和事務沒有半毛錢關系,此時你再調用帶注解的方法,照樣沒有事務,只是一個普通的方法調用而已。
簡單來說,內部調用本類方法,不會再走代理了,所以B的事務不起作用
總結
以上為個人經驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關文章
Spring Web MVC框架學習之配置Spring Web MVC
這一篇文章講的是Spring Web MVC各部分的配置方法,包括Java代碼配置和XML文件配置以及MVC命名空間的使用方法。2017-03-03SpringBoot項目多數據源及mybatis 駝峰失效的問題解決方法
這篇文章主要介紹了SpringBoot項目多數據源及mybatis 駝峰失效的問題解決方法,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-07-07淺談MyBatisPlus中LocalDateTime引發(fā)的一些問題和解決辦法
MyBatisPlus進行數據庫操作時,我們經常會遇到處理日期時間類型的需求,本文主要介紹了淺談MyBatisPlus中LocalDateTime引發(fā)的一些問題和解決辦法,具有一定的參考價值,感興趣的可以了解一下2024-07-07Java?LocalDateTime獲取時間信息、格式化、轉換為數字時間戳代碼示例
其實我們在Java項目中對日期進行格式化,主要是利用一些日期格式化類,下面這篇文章主要給大家介紹了關于Java?LocalDateTime獲取時間信息、格式化、轉換為數字時間戳的相關資料,需要的朋友可以參考下2023-11-11MybatisPlus字段自動填充失效,填充值為null的解決方案
這篇文章主要介紹了MybatisPlus字段自動填充失效,填充值為null的解決方案,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-01-01