Spring 七種事務傳播性介紹

作者:vivo 互聯網服務器團隊 - Zhou Shaobin

本文主要介紹了Spring事務傳播性的相關知識。

Spring中定義了7種事務傳播性:

  • PROPAGATION_REQUIRED 

  • PROPAGATION_SUPPORTS

  • PROPAGATION_MANDATORY

  • PROPAGATION_REQUIRES_NEW

  • PROPAGATION_NOT_SUPPORTED

  • PROPAGATION_NEVER

  • PROPAGATION_NESTED

在Spring環境中,含有事務的方法嵌套調用,事務是如何傳遞的規則,以及每種規則是如何開展工作的。文章還提到每種事務傳播性是如何使用的,方便讀者依據實際的場景,使用不同的事務規則。

一、什麼是Spring事務的傳播性

Spring 事務傳播性是指, 在Spring的環境中,當多個含有事務的方法嵌套調用時,每個事務方法都處於自己事務的上下文中,其提交或者回滾行爲應該如何處理。

通俗講,就是當一個事務方法調用另外一個事務方法時,事務如何跨上下文傳播。

圖片

1)當事務方法A調用事務方法B時,事務方法B是合併到事務方法A中,還是開啓新事務?

2)當事務方法B拋出異常時  ,在合併事務或者開啓新的事務的場景中,事務的回滾是如何處理的 ?

以上事務的處理規則,都取決於事務傳播級別的設置。

二、事務的傳播性都有哪些行爲

圖片

事務的傳播行爲,主要分爲三種類型,分別是:支持當前事務不支持當前事務嵌套事務

2.1 支持當前事務

REQUIRED:默認的事務傳播級別,表示如果當前方法已在事務內,該方法就在當前事務中執行,否則,開啓一個新的事務並在其上下文中執行。

SUPPORTED:當前方法在事務內,則在其上下文中執行該方法,否則,開啓一個新的事務。

MANDATORY:必須在事務中執行,否則,將拋出異常。

2.2 不支持當前事務

REQUIRES_NEW:無論當前是否有事務上下文,都會開啓一個事務  。如果已經有一個事務在執行 ,則正在執行的事務將被掛起 ,新開啓的事務會被執行。

事務之間相互獨立,互不干擾。

NOT_SUPPORTED:不支持事務,如果當前存在事務上下文,則掛起當前事務,然後以非事務的方式執行。

NEVER:不能在事務中執行,如果當前存在事務上下文,則拋出異常。

2.3 嵌套事務

NESTED:嵌套事務,如果當前已存在一個事務的上下文中,則在嵌套事務中執行,如果拋異常,則回滾嵌套事務,而不影響其他事務的操作。

三、每種事務的傳播性如何工作

3.1 REQUIRED  

默認的事務傳播行爲,保證多個嵌套的事務方法在同一個事務內執行,並且同時提交,或者出現異常時,同時回滾。

這個機制可以滿足大多數業務場景。

圖片

例子 :

圖片

圖片

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

1)類TestAService的方法通過聲明式事務的方式,加上了事務註解@Transactional ,並設置事務的傳播性爲REQUIRED。

2)調用者調用TestAService的A方法時,如果調用者沒有開啓事務,那麼A方法會開啓一個事務。

A方法的具體執行過程如下 :

a. 執行insert,但沒有提交;

b. 調用TestBServcie的B方法,由於B方法也聲明瞭事務,並且傳播性是REQUIRED,所以方法B的事務,合併到方法A開啓的事務中。

c. 方法B執行insert操作,此時也沒有提交。

3)由於這兩個方法的操作都在同一個事務中執行,當這兩個方法所有操作執行成功之後,提交事務。

嵌套調用鏈路:

圖片

當方法B 執行時拋出了 Exception 異常後,事務是如何處理的 ?

1)方法B聲明瞭事務,insert操作會回滾

2)由於方法A和方法B 同屬一個事務,方法A也會執行回滾,由此說明該規則保證了事務的原子性。

嵌套調用,異常後的鏈路:

圖片

如果 方法B 拋出異常後,方法A 使用 try-catch 處理了方法B的異常(如下代碼),並沒有向外拋出,此時事務又如何處理的 ?

圖片

 

 

 

 

 

 

 

 

方法A也會回滾。

從事務的特性我們可知,事務具有原子性。方法A和方法B同屬一個事務,當方法B拋出異常,觸發回滾操作後,整個事務的操作都會回滾。

因此,Spring 在處理事務過程中,當事務的傳播性設置爲REQUIRED,在整個事務的調用鏈上,任何一個環節拋出的異常都會導致全局回滾。

3.2 REQUIRES_ NEW

每次都開啓一 個新的事務。

圖片

例子:

圖片

 

 

 

 

 

上面例子中,方法B的傳播性設置爲 REQUIRES_NEW,方法A仍然是REQUIRED,當A調用B時,具體調用鏈路如下:

圖片

具體執行過程:

  • 方法A被執行前,如果調用者沒有開啓事務,方法A開啓一個事務1,然後執行insert ,此時沒有提交;

  • 方法B的事務傳播性設置爲REQUIRES_NEW,當被方法A調用時,此時方法A的事務1會被掛起,方法B開啓自己的事務2,然後執行insert,此時並沒有提交;

  • 當方法B執行完畢後,提交事務2;

  • 恢復事務1,最終提交。

當 方法B 執行時拋出了異常,會發生什麼?

方法B的insert操作會被回滾掉,方法A不受影響。但這裏有個前提,方法A需要try-catch方法B的異常,使其異常不會往上傳遞,從而導致方法A接收到異常,導致回滾。

圖片

3.3  SUPPORTED

當外層方法A存在事務,方法B加入到當前事務中,以事務的方式執行。

圖片

當外層方法A不存在事務,方法B不會創建新的事務,以非事務的方式執行。

圖片

例子1:

圖片

 

 

 

 

 

圖片

以上例子,方法A沒有加事務註解,方法B的加了事務註解,並且傳播爲SUPPORTS。

具體執行過程:

  • 方法A以非事務的方式執行insert操作。

  • 方法B被調用,由於其外層事務A沒有開啓事務,方法B也是以非事務方法執行insert操作。

圖片

例子2:

圖片

以上例子,方法A和B都加上了事務註解,其中方法A的傳播性爲REQUIRED,方法B的傳播性爲SUPPORTS。

具體執行過程:

  • 如果方法A的調用方沒有開啓事務,則方法A開啓事務,並執行insert操作,但沒有提交;

  • 方法B被調用,由於其外層方法A開啓了事務,因此方法B加入到方法A開啓的事務中,並執行insert,但沒有提交;

  • 當事務中的所有操作執行成功後,事務提交。

圖片

3.4  NOT_SUPPORTED

不支持事務。

如果外層方法存在事務,則掛起外層事務,以非事務方式執行,執行完畢後,恢復外層事務。

圖片

例子:

圖片

以上例子:方法A和B都加上了事務註解,方法A的傳播性爲REQUIRED,方法B爲NOT_SUPPORTED。

具體執行過程:

  • 如A的調用方沒有開啓事務,方法A開啓事務,並執行insert,但沒有提交。

  • 方法A調用方法B時,方法B的傳播性爲NOT_SUPPORTED,不支持事務,然後掛起外層方法A的事務,方法B以非事務的方式執行insert。

  • 方法B執行完畢後,恢復方法A的事務,最終提交事務。

調用鏈路過程:

圖片

3.5 NEVER

不支持事務

當外層方法A開啓了事務,方法B拋出異常

圖片

例子:

圖片

以上代碼,兩個方法都打上了事務註解,方法A的傳播性是REQUIRED,方法B的傳播性是NEVER。

具體執行過程:

  • 方法A開啓事務,執行insert,沒有提交。

  • 含有事務的方法A調用方法B,方法B的傳播性是NEVER,表示不支持事務,因此方法B拋出異常。

  • 方法A的事務執行回滾。

圖片

3.6 MANDATORY

必須在事務中執行。

如果外層方法A沒有開啓事務,方法B拋出異常。

圖片

如果外層方法A開啓了事務,方法B加入事務,方法A&B在同一事務中執行。

圖片

例子:

圖片

以上例子,方法A沒有加事務註解,方法B 的傳播性爲 MANDATORY。

具體執行過程:

  • 方法A的調用方如果本身沒有開啓事務,方法A執行前不會開啓事務。

  • 當非事務方法A調用方法B時,由於方法B的傳播性爲MANDATORY,必須在事務中執行,條件不滿足,拋出異常。

圖片

3.7 NESTED

嵌套事務

  • 如果外層方法A不存在事務,內層方法B的規則與REQUIRED 一致。

  • 如果外層方法A存在事務,內層方法B做爲外層方法A事務的子事務執行,兩個方法是一起提交,但子事務是獨立回滾。

    內層方法B拋出異常,則會回滾方法B的所有操作,但不影響外層事務方法A。(方法A需要try-catch子事務,避免異常傳遞到父層事務)

    外層方法A回滾,則內層方法B也會回滾。

  • 該傳播性的特點是可以保存狀態點,當回滾時,只會回滾到某一個狀態點,保證了子事務之間的獨立性,避免嵌套事務的全局回滾。

圖片

例子:

圖片

以上例子,方法A的傳播性爲REQUIRED,方法B爲NESTED。

具體執行過程:

  • 方法A執行時,如調用方沒有開啓事務,則開啓一個事務。

  • 方法B被外層方法A調用時,因爲方法B的傳播性爲NESTED,方法B在此處建立savepoint,標記insert行爲。

  • 當方法B拋出異常,其insert操作會回滾,但只會回滾到savepoint,(前提是方法A要try-catch方法B,使方法B的異常不會往外傳遞)。

  • 方法B回滾後,方法A的事務提交。

調用鏈路:

圖片

四、總結

本文解釋了Spring框架中的事務傳播性,即多個業務方法之間調用時事務如何處理的規則。Spring提供了七種傳播級別,如PROPAGATION_REQUIRED、PROPAGATION_REQUIRES_NEW等。

每種級別都有適用場景和限制,本文提供了一些示例,介紹了聲明式事務如何使用,每種事務的規則,產生哪種行爲,當方法拋出異常時,事務的提交和回滾是如何被處理的。正確處理事務對於任何企業級應用程序都是必要的,瞭解Spring事務傳播性是構建高效、可靠和可擴展應用程序的關鍵。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章