Spring在TransactionDefinition接口中規定了7種類型的事務傳播行爲。下面這篇文章主要給大家介紹了關於Spring事務傳播行爲的相關資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面來一起學習學習吧
前言
本文主要介紹下Spring事務中的傳播行爲。事務傳播行爲是Spring框架獨有的事務增強特性,他不屬於的事務實際提供方數據庫行爲。這是Spring爲我們提供的強大的工具箱,使用事務傳播行可以爲我們的開發工作提供許多便利。
下面話不多說了,來一起看看詳細的介紹吧
事務傳播行爲介紹
Spring中的7個事務傳播行爲:
|事務行爲|說明 |
|:--|:--|
|PROPAGATION_REQUIRED | 支持當前事務,假設當前沒有事務。就新建一個事務 |
| PROPAGATION_SUPPORTS |支持當前事務,假設當前沒有事務,就以非事務方式運行 |
| PROPAGATION_MANDATORY| 支持當前事務,假設當前沒有事務,就拋出異常|
| PROPAGATION_REQUIRES_NEW | 新建事務,假設當前存在事務。把當前事務掛起|
|PROPAGATION_NOT_SUPPORTED | 以非事務方式運行操作。假設當前存在事務,就把當前事務掛起 |
| PROPAGATION_NEVER | 以非事務方式運行,假設當前存在事務,則拋出異常 |
| PROPAGATION_NESTED |如果當前存在事務,則在嵌套事務內執行。如果當前沒有事務,則執行與PROPAGATION_REQUIRED類似的操作。 |
舉例說明
案例代碼
ServiceA
ServiceA { void methodA() { ServiceB.methodB(); } }
ServiceB
ServiceB { void methodB() { } }
1.PROPAGATION_REQUIRED
假如當前正要運行的事務不在另外一個事務裏,那麼就起一個新的事務 比方說,ServiceB.methodB的事務級別定義PROPAGATION_REQUIRED, 那麼因爲執行ServiceA.methodA的時候,ServiceA.methodA已經起了事務。這時調用ServiceB.methodB,ServiceB.methodB看到自己已經執行在ServiceA.methodA的事務內部。就不再起新的事務。而假如ServiceA.methodA執行的時候發現自己沒有在事務中,他就會爲自己分配一個事務。這樣,在ServiceA.methodA或者在ServiceB.methodB內的不論什麼地方出現異常。事務都會被回滾。即使ServiceB.methodB的事務已經被提交,可是ServiceA.methodA在接下來fail要回滾,ServiceB.methodB也要回滾
2.PROPAGATION_SUPPORTS
假設當前在事務中。即以事務的形式執行。假設當前不在一個事務中,那麼就以非事務的形式執行
3PROPAGATION_MANDATORY
必須在一個事務中執行。也就是說,他僅僅能被一個父事務調用。否則,他就要拋出異常
4.PROPAGATION_REQUIRES_NEW
這個就比較繞口了。 比方我們設計ServiceA.methodA的事務級別爲PROPAGATION_REQUIRED,ServiceB.methodB的事務級別爲PROPAGATION_REQUIRES_NEW。那麼當運行到ServiceB.methodB的時候,ServiceA.methodA所在的事務就會掛起。ServiceB.methodB會起一個新的事務。等待ServiceB.methodB的事務完畢以後,他才繼續運行。
他與PROPAGATION_REQUIRED 的事務差別在於事務的回滾程度了。由於ServiceB.methodB是新起一個事務,那麼就是存在兩個不同的事務。假設ServiceB.methodB已經提交,那麼ServiceA.methodA失敗回滾。ServiceB.methodB是不會回滾的。假設ServiceB.methodB失敗回滾,假設他拋出的異常被ServiceA.methodA捕獲,ServiceA.methodA事務仍然可能提交。
5.PROPAGATION_NOT_SUPPORTED
當前不支持事務。比方ServiceA.methodA的事務級別是PROPAGATION_REQUIRED 。而ServiceB.methodB的事務級別是PROPAGATION_NOT_SUPPORTED ,那麼當執行到ServiceB.methodB時。ServiceA.methodA的事務掛起。而他以非事務的狀態執行完,再繼續ServiceA.methodA的事務。
6.PROPAGATION_NEVER
不能在事務中執行。
如果ServiceA.methodA的事務級別是PROPAGATION_REQUIRED。 而ServiceB.methodB的事務級別是PROPAGATION_NEVER ,那麼ServiceB.methodB就要拋出異常了。
7.PROPAGATION_NESTED
如果當前存在事務,則在嵌套事務內執行。如果當前沒有事務,則執行與PROPAGATION_REQUIRED類似的操作。
Spring中事務的配置
配置文件的方式
<tx:advice id="txAdvice" transaction-manager="txManager"> <tx:attributes> <!--設置所有匹配的方法,然後設置傳播級別和事務隔離--> <tx:method name="save*" propagation="REQUIRED" /> <tx:method name="add*" propagation="REQUIRED" /> <tx:method name="create*" propagation="REQUIRED" /> <tx:method name="insert*" propagation="REQUIRED" /> <tx:method name="update*" propagation="REQUIRED" /> <tx:method name="merge*" propagation="REQUIRED" /> <tx:method name="del*" propagation="REQUIRED" /> <tx:method name="remove*" propagation="REQUIRED" /> <tx:method name="put*" propagation="REQUIRED" /> <tx:method name="get*" propagation="SUPPORTS" read-only="true" /> <tx:method name="count*" propagation="SUPPORTS" read-only="true" /> <tx:method name="find*" propagation="SUPPORTS" read-only="true" /> <tx:method name="list*" propagation="SUPPORTS" read-only="true" /> <tx:method name="*" propagation="SUPPORTS" read-only="true" /> </tx:attributes> </tx:advice>
註解的方式
<!--開啓註解的方式--> <tx:annotation-driven transaction-manager="transactioManager" />
@Transactional(propagation=Propagation.REQUIRED)
如果有事務, 那麼加入事務, 沒有的話新建一個(默認情況下)
@Transactional(propagation=Propagation.NOT_SUPPORTED)
容器不爲這個方法開啓事務
@Transactional(propagation=Propagation.REQUIRES_NEW)
不管是否存在事務,都創建一個新的事務,原來的掛起,新的執行完畢,繼續執行老的事務
@Transactional(propagation=Propagation.MANDATORY)
必須在一個已有的事務中執行,否則拋出異常
@Transactional(propagation=Propagation.NEVER)
必須在一個沒有的事務中執行,否則拋出異常(與Propagation.MANDATORY相反)
@Transactional(propagation=Propagation.SUPPORTS)
如果其他bean調用這個方法,在其他bean中聲明事務,那就用事務.如果其他bean沒有聲明事務,那就不用事務.
總結
以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對神馬文庫的支持。