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。

推薦閱讀: