springboot多個(gè)service互相調(diào)用的事務(wù)處理方式
多個(gè)service互相調(diào)用的事務(wù)處理
今天,想在一個(gè)service的方法A中,調(diào)用另一個(gè)service的方法B,方法A和方法B均存在數(shù)據(jù)庫(kù)插入操作,并且@Transaction注解也都加了,但是當(dāng)B方法中拋出異常時(shí),A中的插入語(yǔ)句還是能夠執(zhí)行成功。
注解配置如下
@Transactional(isolation= Isolation.DEFAULT,propagation= Propagation.REQUIRED)
百思不得其解,再查找了相關(guān)資料后,問(wèn)題還是出在@Transaction注解的配置上,需要配置異?;貪L。
@Transactional(isolation= Isolation.DEFAULT,propagation= Propagation.REQUIRED,rollbackFor = Exception.class)
這樣,當(dāng)B方法中拋出異常時(shí),A中的操作也會(huì)進(jìn)行回滾,事務(wù)就會(huì)起到控制作用。
Spring事務(wù)調(diào)用Service和Service之間的調(diào)用
同一個(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() { ? ? ? ? ? ? //開(kāi)啟事務(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(); ? ? ? ? } ? ? }
那回到一開(kāi)始的問(wèn)題,我們調(diào)用的方法A不帶注解,因此代理類不開(kāi)事務(wù),而是直接調(diào)用目標(biāo)對(duì)象的方法。
當(dāng)進(jìn)入目標(biāo)對(duì)象的方法后,執(zhí)行的上下文已經(jīng)變成目標(biāo)對(duì)象本身了,因?yàn)槟繕?biāo)對(duì)象的代碼是我們自己寫(xiě)的,和事務(wù)沒(méi)有半毛錢(qián)關(guān)系,此時(shí)你再調(diào)用帶注解的方法,照樣沒(méi)有事務(wù),只是一個(gè)普通的方法調(diào)用而已。
簡(jiǎn)單來(lái)說(shuō),內(nèi)部調(diào)用本類方法,不會(huì)再走代理了,所以B的事務(wù)不起作用
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
SpringBoot利用自定義注解實(shí)現(xiàn)隱私數(shù)據(jù)脫敏(加密顯示)的解決方案
這兩天在整改等保測(cè)出的問(wèn)題,里面有一個(gè)“用戶信息泄露”的風(fēng)險(xiǎn)項(xiàng)(就是后臺(tái)系統(tǒng)里用戶的一些隱私數(shù)據(jù)直接明文顯示了),其實(shí)指的就是要做數(shù)據(jù)脫敏,本文給大家介紹了SpringBoot利用自定義注解實(shí)現(xiàn)隱私數(shù)據(jù)脫敏(加密顯示)的解決方案,需要的朋友可以參考下2023-11-11spring boot+thymeleaf+bootstrap實(shí)現(xiàn)后臺(tái)管理系統(tǒng)界面
這篇文章主要為大家詳細(xì)介紹了spring boot+thymeleaf+bootstrap簡(jiǎn)單實(shí)現(xiàn)后臺(tái)管理系統(tǒng)界面,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-12-12pom.xml中解決Provides?transitive?vulnerable?dependency?mave
這篇文章主要介紹了在pom.xml中如何解決Provides?transitive?vulnerable?dependency?maven:org.yaml:snakeyaml:1.33警告問(wèn)題,需要的朋友可以參考下2023-06-06通過(guò)Java帶你了解網(wǎng)絡(luò)IO模型
這篇文章將通過(guò)Java帶大家了解網(wǎng)絡(luò)IO模型,包括BIO,NoBlockingIO,NIO(NewIO),AIO等做了詳細(xì)得介紹,感興趣的小伙伴可以參考閱讀本文2023-05-05SpringBoot接口返回結(jié)果封裝方法實(shí)例詳解
在實(shí)際項(xiàng)目中,一般會(huì)把結(jié)果放在一個(gè)封裝類中,封裝類中包含http狀態(tài)值,狀態(tài)消息,以及實(shí)際的數(shù)據(jù)。這里主要記錄兩種方式,通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),需要的朋友參考下吧2021-09-09