springboot多個service互相調用的事務處理方式
多個service互相調用的事務處理
今天,想在一個service的方法A中,調用另一個service的方法B,方法A和方法B均存在數(shù)據(jù)庫插入操作,并且@Transaction注解也都加了,但是當B方法中拋出異常時,A中的插入語句還是能夠執(zhí)行成功。
注解配置如下
@Transactional(isolation= Isolation.DEFAULT,propagation= Propagation.REQUIRED)
百思不得其解,再查找了相關資料后,問題還是出在@Transaction注解的配置上,需要配置異?;貪L。
@Transactional(isolation= Isolation.DEFAULT,propagation= Propagation.REQUIRED,rollbackFor = Exception.class)
這樣,當B方法中拋出異常時,A中的操作也會進行回滾,事務就會起到控制作用。
Spring事務調用Service和Service之間的調用
同一個類的不同方法,A方法沒有@Transactional,B方法有@Transactional,A調用B方法,事務不起作用
原理解析
spring 在掃描bean的時候會掃描方法上是否包含@Transactional注解,如果包含,spring會為這個bean動態(tài)地生成一個子類(即代理類,proxy),代理類是繼承原來那個bean的。
此時,當這個有注解的方法被調用的時候,實際上是由代理類來調用的,代理類在調用之前就會啟動transaction。
然而,如果這個有注解的方法是被同一個類中的其他方法調用的,那么該方法的調用并沒有通過代理類,而是直接通過原來的那個bean,所以就不會啟動transaction,我們看到的現(xiàn)象就是@Transactional注解無效。
? ? //接口 ? ? interface Service { ? ? ? ? void A();? ?? ? ? ? ? void B(); ? ? } ? ?? ? ? //目標類,實現(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"); ? ? ? ? } ? ? } ? ?? ? ? //代理類,也要實現(xiàn)相同的接口 ? ? 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í)行的上下文已經(jīng)變成目標對象本身了,因為目標對象的代碼是我們自己寫的,和事務沒有半毛錢關系,此時你再調用帶注解的方法,照樣沒有事務,只是一個普通的方法調用而已。
簡單來說,內(nèi)部調用本類方法,不會再走代理了,所以B的事務不起作用
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關文章
SpringBoot利用自定義注解實現(xiàn)隱私數(shù)據(jù)脫敏(加密顯示)的解決方案
這兩天在整改等保測出的問題,里面有一個“用戶信息泄露”的風險項(就是后臺系統(tǒng)里用戶的一些隱私數(shù)據(jù)直接明文顯示了),其實指的就是要做數(shù)據(jù)脫敏,本文給大家介紹了SpringBoot利用自定義注解實現(xiàn)隱私數(shù)據(jù)脫敏(加密顯示)的解決方案,需要的朋友可以參考下2023-11-11spring boot+thymeleaf+bootstrap實現(xiàn)后臺管理系統(tǒng)界面
這篇文章主要為大家詳細介紹了spring boot+thymeleaf+bootstrap簡單實現(xiàn)后臺管理系統(tǒng)界面,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-12-12pom.xml中解決Provides?transitive?vulnerable?dependency?mave
這篇文章主要介紹了在pom.xml中如何解決Provides?transitive?vulnerable?dependency?maven:org.yaml:snakeyaml:1.33警告問題,需要的朋友可以參考下2023-06-06