springboot多個service互相調用的事務處理方式
多個service互相調用的事務處理
今天,想在一個service的方法A中,調用另一個service的方法B,方法A和方法B均存在數據庫插入操作,並且@Transaction註解也都加瞭,但是當B方法中拋出異常時,A中的插入語句還是能夠執行成功。
註解配置如下
@Transactional(isolation= Isolation.DEFAULT,propagation= Propagation.REQUIRED)
百思不得其解,再查找瞭相關資料後,問題還是出在@Transaction註解的配置上,需要配置異常回滾。
@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動態地生成一個子類(即代理類,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不帶註解,因此代理類不開事務,而是直接調用目標對象的方法。
當進入目標對象的方法後,執行的上下文已經變成目標對象本身瞭,因為目標對象的代碼是我們自己寫的,和事務沒有半毛錢關系,此時你再調用帶註解的方法,照樣沒有事務,隻是一個普通的方法調用而已。
簡單來說,內部調用本類方法,不會再走代理瞭,所以B的事務不起作用
以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。
推薦閱讀:
- springBoot service層事務控制的操作
- SpringBoot在自定義類中調用service層等Spring其他層操作
- Java spring事務及事務不生效的原因詳解
- springboot中使用@Transactional註解事物不生效的坑
- Spring中propagation的7種事務配置及說明