這篇文章給大家分享的是有關spring事務Propagation怎么實現(xiàn)的內(nèi)容。小編覺得挺實用的,因此分享給大家做個參考,一起跟隨小編過來看看吧。
10年的蘇州網(wǎng)站建設經(jīng)驗,針對設計、前端、開發(fā)、售后、文案、推廣等六對一服務,響應快,48小時及時工作處理。營銷型網(wǎng)站的優(yōu)勢是能夠根據(jù)用戶設備顯示端的尺寸不同,自動調(diào)整蘇州建站的顯示方式,使網(wǎng)站能夠適用不同顯示終端,在瀏覽器中調(diào)整網(wǎng)站的寬度,無論在任何一種瀏覽器上瀏覽網(wǎng)站,都能展現(xiàn)優(yōu)雅布局與設計,從而大程度地提升瀏覽體驗。創(chuàng)新互聯(lián)公司從事“蘇州網(wǎng)站設計”,“蘇州網(wǎng)站推廣”以來,每個客戶項目都認真落實執(zhí)行。具體介紹如下。
簡介spring目前已是java開發(fā)的一個事實標準,這得益于它的便利、功能齊全、容易上手等特性。在開發(fā)過程當中,操作DB是非常常見的操作,而涉及到db,就會涉及到事務。事務在平時的開發(fā)過程當中,就算沒有注意到,程序正常執(zhí)行不會有副作用,但如果出現(xiàn)了異常,而又沒有處理好事務的話,可能就會出現(xiàn)意想不到的結果。spring在事務方面進行了各種操作的封裝,特別是聲明式事務的出現(xiàn),讓開發(fā)變得更加的舒心。spring對事務進行了擴展,支持定義多種傳播屬性,這也是本篇要說明的重點。
事務是什么非嚴格的講,一個事務是多個操作的簡稱,這些操作要么全部生效,要么一個都不生效(相當于沒有執(zhí)行過),一個通用的操作流程簡化如下:
try{ Connection conn = getConnection(); // 執(zhí)行一些數(shù)據(jù)庫操作 }catch(Exception e){ conn.rollback(); }finally{ conn.close(); }
從以上代碼可以看出一些問題:
太多無用的固定代碼
如果一個請求需要調(diào)用多個服務接口,難以更精細的控制事務
跨多種底層數(shù)據(jù)層,如jdbc,mybatis,hibernate,jta,難以統(tǒng)一編碼方式。
spring提供了聲明式事務,使得我們不用關注底層的具體實現(xiàn),屏蔽了多種不同的底層實現(xiàn)細節(jié),為了支持多種復雜業(yè)務對事務的精細控制,spring提供了事務的傳播屬性,結合聲明式事務,成就了一大事務利器。
spring事務傳播屬性示例分析在TransactionDefinition類中,spring提供了6種傳播屬性,接下來分別用簡單示例來說明。
溫馨提醒:下文提到的加入當前事務,指的是底層使用同一個Connection,但是事務狀態(tài)對象是可以重新創(chuàng)建的,并不影響。文章提到的當前只存在一個事務,表示的是共用底層的一個Connection,而不在乎創(chuàng)建了多少個事務狀態(tài)對象(TransactionStatus)。
1、PROPAGATION_REQUIRED說明: 如果當前已經(jīng)存在事務,那么加入該事務,如果不存在事務,創(chuàng)建一個事務,這是默認的傳播屬性值。
看一個小例子,代碼如下:
@Transactional public void service(){ serviceA(); serviceB(); } @Transactional serviceA(); @Transactional serviceB();
serviceA 和 serviceB 都聲明了事務,默認情況下,propagation=PROPAGATION_REQUIRED,整個service調(diào)用過程中,只存在一個共享的事務,當有任何異常發(fā)生的時候,所有操作回滾。
2、PROPAGATION_SUPPORTS說明:如果當前已經(jīng)存在事務,那么加入該事務,否則創(chuàng)建一個所謂的空事務(可以認為無事務執(zhí)行)。
看一個小例子,代碼如下:
public void service(){ serviceA(); throw new RunTimeException(); } @Transactional(propagation=Propagation.SUPPORTS) serviceA();
serviceA執(zhí)行時當前沒有事務,所以service中拋出的異常不會導致 serviceA回滾。
再看一個小例子,代碼如下:
public void service(){ serviceA(); } @Transactional(propagation=Propagation.SUPPORTS) serviceA(){ do sql 1 1/0; do sql 2 }
由于serviceA運行時沒有事務,這時候,如果底層數(shù)據(jù)源defaultAutoCommit=true,那么sql1是生效的,如果defaultAutoCommit=false,那么sql1無效,如果service有@Transactional標簽,serviceA共用service的事務(不再依賴defaultAutoCommit),此時,serviceA全部被回滾。
3、 PROPAGATION_MANDATORY說明:當前必須存在一個事務,否則拋出異常。
看一個小例子,代碼如下:
public void service(){ serviceB(); serviceA(); } serviceB(){ do sql } @Transactional(propagation=Propagation.MANDATORY) serviceA(){ do sql }
這種情況執(zhí)行 service會拋出異常,如果defaultAutoCommit=true,則serviceB是不會回滾的,defaultAutoCommit=false,則serviceB執(zhí)行無效。
4、PROPAGATN_REQUIRES_NEW說明:如果當前存在事務,先把當前事務相關內(nèi)容封裝到一個實體,然后重新創(chuàng)建一個新事務,接受這個實體為參數(shù),用于事務的恢復。更直白的說法就是暫停當前事務(當前無事務則不需要),創(chuàng)建一個新事務。 針對這種情況,兩個事務沒有依賴關系,可以實現(xiàn)新事務回滾了,但外部事務繼續(xù)執(zhí)行。
看一個小例子,代碼如下:
@Transactional public void service(){ serviceB(); try{ serviceA(); }catch(Exception e){ } } serviceB(){ do sql } @Transactional(propagation=Propagation.REQUIRES_NEW) serviceA(){ do sql 1 1/0; do sql 2 }
當調(diào)用service接口時,由于serviceA使用的是REQUIRES_NEW,它會創(chuàng)建一個新的事務,但由于serviceA拋出了運行時異常,導致serviceA整個被回滾了,而在service方法中,捕獲了異常,所以serviceB是正常提交的。 注意,service中的try … catch 代碼是必須的,否則service也會拋出異常,導致serviceB也被回滾。
5、Propagation.NOT_SUPPORTED說明:如果當前存在事務,掛起當前事務,然后新的方法在沒有事務的環(huán)境中執(zhí)行,沒有spring事務的環(huán)境下,sql的提交完全依賴于 defaultAutoCommit屬性值 。
看一個小例子,代碼如下:
@Transactional public void service(){ serviceB(); serviceA(); } serviceB(){ do sql } @Transactional(propagation=Propagation.NOT_SUPPORTED) serviceA(){ do sql 1 1/0; do sql 2 }
當調(diào)用service方法的時候,執(zhí)行到serviceA方法中的1/0代碼時,拋出了異常,由于serviceA處于無事務環(huán)境下,所以 sql1是否生效取決于defaultAutoCommit的值,當defaultAutoCommit=true時,sql1是生效的,但是service由于拋出了異常,所以serviceB會被回滾。
6、 PROPAGATION_NEVER說明: 如果當前存在事務,則拋出異常,否則在無事務環(huán)境上執(zhí)行代碼。
看一個小例子,代碼如下:
public void service(){ serviceB(); serviceA(); } serviceB(){ do sql } @Transactional(propagation=Propagation.NEVER) serviceA(){ do sql 1 1/0; do sql 2 }
上面的示例調(diào)用service后,若defaultAutoCommit=true,則serviceB方法及serviceA中的sql1都會生效。
7、 PROPAGATION_NESTED說明: 如果當前存在事務,則使用 SavePoint 技術把當前事務狀態(tài)進行保存,然后底層共用一個連接,當NESTED內(nèi)部出錯的時候,自行回滾到 SavePoint這個狀態(tài),只要外部捕獲到了異常,就可以繼續(xù)進行外部的事務提交,而不會受到內(nèi)嵌業(yè)務的干擾,但是,如果外部事務拋出了異常,整個大事務都會回滾。
注意: spring配置事務管理器要主動指定 nestedTransactionAllowed=true,如下所示:
看一個小例子,代碼如下:
@Transactional public void service(){ serviceA(); try{ serviceB(); }catch(Exception e){ } } serviceA(){ do sql } @Transactional(propagation=Propagation.NESTED) serviceB(){ do sql1 1/0; do sql2 }
serviceB是一個內(nèi)嵌的業(yè)務,內(nèi)部拋出了運行時異常,所以serviceB整個被回滾了,由于service捕獲了異常,所以serviceA是可以正常提交的。
再來看一個例子,代碼如下:
@Transactional public void service(){ serviceA(); serviceB(); 1/0; } @Transactional(propagation=Propagation.NESTED) serviceA(){ do sql } serviceB(){ do sql }
由于service拋出了異常,所以會導致整個service方法被回滾。(這就是跟PROPAGATION_REQUIRES_NEW不一樣的地方了,NESTED方式下的內(nèi)嵌業(yè)務會受到外部事務的異常而回滾。)
實現(xiàn)淺析前面舉例說明了spring事務提供的幾種傳播屬性,用于滿足多種不同的業(yè)務需求,大家可以依業(yè)務而定。接著我們再來看看spring實現(xiàn)這些傳播屬性最重要的技術依賴是什么。本小節(jié)列舉 PROPAGATION_REQUIRES_NEW 和 Propagation.NESTED 分別進行簡要說明。
1、 PROPAGATION_REQUIRES_NEW 實現(xiàn)原理如下的代碼調(diào)用:
@Transactional public void service(){ serviceB(); try{ serviceA(); }catch(Exception e){ } } @Transactional(propagation=Propagation.REQUIRES_NEW) serviceA(){ do sql 1 1/0; do sql 2 } serviceB(){ do sql }
執(zhí)行原理圖如下:
a. 創(chuàng)建事務狀態(tài)對象,獲取一個新的連接,重置連接的 autoCommit,fetchSize,timeout等屬性
b. 把連接綁定到ThreadLocal變量
c. 掛起當前事務,把當前事務狀態(tài)對象,連接等信息封裝成一SuspendedResources對象,可用于恢復
d. 創(chuàng)建新的事務狀態(tài)對象,重新獲取新的連接,重置新連接的 autoCommit,fetchSize,timeout等屬性,同時,保存SuspendedResources對象,用于事務的恢復,把新的連接綁定到ThreadLocal變量(覆蓋操作)
e. 捕獲到異常,回滾ThreadLocal中的連接,恢復連接參數(shù),關閉連接,恢復SuspendedResources
f. 提交ThreadLocal變量中的連接(導致serviceB被提交),還原連接參數(shù),關閉連接,連接歸還數(shù)據(jù)源
所以程序執(zhí)行的結果就是 serviceA被回滾了,serviceB成功提交了。
2、 PROPAGATION_NESTED 實現(xiàn)原理如下的代碼調(diào)用:
@Transactional public void service(){ serviceA(); try{ serviceB(); }catch(Exception e){ } } serviceA(){ do sql } @Transactional(propagation=Propagation.NESTED) serviceB(){ do sql1 1/0; do sql2 }
執(zhí)行原理圖如下:
a. 創(chuàng)建事務狀態(tài)對象,獲取一個新的連接,重置連接的 autoCommit,fetchSize,timeout等屬性
b. 把連接綁定到ThreadLocal變量
c. 標記使用當前事務狀態(tài)對象,獲取ThreadLocal連接對象,保存當前連接的SavePoint,用于異?;謴?,此時的SavePoint就是執(zhí)行完serviceA后的狀態(tài)
d. 捕獲到異常,使用c中的SavePoint進行事務回滾,也就是把狀態(tài)回滾到執(zhí)行serviceA后的狀態(tài),serviceB方法所有執(zhí)行不生效
e. 獲取ThreadLocal中的連接對象,提交事務,恢復連接屬性,關閉連接
其它spring在底層數(shù)據(jù)源的基礎上,利用 ThreadLocal,SavePoint等技術點實現(xiàn)了多種事務傳播屬性,便于實現(xiàn)各種復雜的業(yè)務。只有理解了傳播屬性的原理才能更好的駕馭spring事務。Spring回滾事務依賴于對異常的捕獲,默認情況下,只有拋出RuntimeException和Error才會回滾事務,當然可以進行配置,更多信息可以查看 @Transactional 這個注解。
感謝各位的閱讀!關于“spring事務Propagation怎么實現(xiàn)”這篇文章就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,讓大家可以學到更多知識,如果覺得文章不錯,可以把它分享出去讓更多的人看到吧!