Spring 如何在一個事務中開啟另一個事務?


作者:Mazin
https://my.oschina.net/u/3441184/blog/893628

這樣的情景可能不常見,但是還是會有的,一旦遇到,如果業務比較復雜,就會很麻煩,但是還是有解決的方案的,比如將一個service方法拆成兩個方法,也就是將兩個操作的事務分開。

但是這只適用與業務比較簡單的,如果出現多次數據庫的寫操作,而我們調用的系統只需要其中一個寫操作的最新數據,如果我們將它分開,那么如果調用目標系統出現異常的時候,那么之前的寫操作就不能回滾了。

舉個簡單的例子:

@Service
public class ServiceA {

  @Transactional
  public void doSomething(){

    向數據庫中添加數據;

    調用其他系統;
  }
}

這里就用偽代碼來做示例了,當我們執行了“向數據庫中添加數據”,我們去數據庫中查詢,發現並沒有我們添加的數據,但是當我們的service這個方法執行完成之后,數據庫中就有這條數據了,這是由於數據庫的隔離性造成的。

Spring中的事務注解 @transactional 提供了一個參數:

Propagation propagation() default Propagation.REQUIRED;

這個參數是定義 Spring事務的傳遞性的,默認值為:required,也就是如果有事務,就加入事務,如果沒有,就創建事務。

這個參數的值有很多,例如:REQUIRES_NEW,這個值就代表創建一個新的事務,與原來的事務分開。這個好像能解決我們的問題。關注Java技術棧公眾號在后台回復:spring,可獲取一份棧長整理的最新 Spring 系列技術干貨。

我們將剛剛那個方法修改一下:

@Service
public class ServiceA {

  @Transactional
  public void doSomething(){

    insert();

    調用其他系統;
  }

  @Transactional(propagation = Propagation.REQUIRES_NEW)
  public void insert(){
    向數據庫中添加數據;
  }
}

執行之后,發現結果還是沒有改變,必須要整體執行完成,數據庫中數據才會出現,說明還是在一個事務中。Spring的核心思想,推薦你看看。

我們再將代碼修改一下:

@Service
public class ServiceA {

  @Autowired
  private ServiceB serviceB;
  @Transactional
  public void doSomething(){

    serviceB.insert();

    調用其他系統;
  }
}
@Service
public class ServiceB {

  @Transactional(propagation = Propagation.REQUIRES_NEW)
  public void insert(){
    向數據庫中添加數據;
  }
}

我們將要事務分離出來的方法寫在另一個service中,再次測試,發現執行完插入語句之后,數據庫中就已經能查到數據了,說明事務分離了,完成了我們的需求。

當然** Spring**其實也考慮這個,在 Spring的配置中,我們只需要添加標簽:

aop:aspectj-autoproxy expose-proxy="true"/

或者:

aop:config expose-proxy="true"

並且在代碼的調用中要求使用代理對象去調用即可:

((ServiceA ) AopContext.currentProxy()).insert();

關注公眾號Java技術棧回復"面試"獲取我整理的2020最全面試題及答案。

推薦去我的博客閱讀更多:

1.Java JVM、集合、多線程、新特性系列教程

2.Spring MVC、Spring Boot、Spring Cloud 系列教程

3.Maven、Git、Eclipse、Intellij IDEA 系列工具教程

4.Java、后端、架構、阿里巴巴等大廠最新面試題

覺得不錯,別忘了點贊+轉發哦!


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM