Spring學(xué)習(xí)JdbcTemplate數(shù)據(jù)庫事務(wù)參數(shù)
Spring JdbcTemplate數(shù)據(jù)庫事務(wù)參數(shù)
@Transactional() 注解里有不少參數(shù),其中我們常用到的如下:
一、propagation
表示事務(wù)傳播行為。就是說多個(gè)事務(wù)方法之間進(jìn)行調(diào)用,這個(gè)過程中事務(wù)是如何進(jìn)行管理的。
這里的事務(wù)方法就是指對數(shù)據(jù)庫表數(shù)據(jù)進(jìn)行變化操作的方法。
舉例:
有個(gè) update() 方法:
public void update() { }
還有個(gè) add() 方法:
public void add() { // 調(diào)用了update()方法 update() }
那么,現(xiàn)在當(dāng)其中一個(gè)方法加上了事務(wù)注解 @Transactional 后,調(diào)用執(zhí)行過程是怎樣的?或者說兩個(gè)都加了事務(wù)注解,又該如何?
為了解決問題,spring 事務(wù)傳播行為有 7 種:
- REQUIRED:默認(rèn)參數(shù)。如果有事務(wù)在運(yùn)行,當(dāng)前方法就在這個(gè)事務(wù)內(nèi)運(yùn)行,否則就開啟一個(gè)新的事務(wù),并在自己的事務(wù)內(nèi)運(yùn)行。
- REQUIRES_NEW:當(dāng)前的方法必須在啟動(dòng)新事務(wù),并在它自己的事務(wù)內(nèi)運(yùn)行,如果有事務(wù)在進(jìn)行,應(yīng)該將它掛起。
- SUPPORTS:如果有事務(wù)在運(yùn)行,當(dāng)前的方法就在這個(gè)事務(wù)內(nèi)運(yùn)行,否則它可以不運(yùn)行在事務(wù)中。
- NOT_SUPPORTED:當(dāng)前的方法不應(yīng)該運(yùn)行在事務(wù)中,如果有運(yùn)行的事務(wù),將它掛起。
- MANDATORY:當(dāng)前的方法必須運(yùn)行在事務(wù)內(nèi)部,如果沒有正在運(yùn)行的事務(wù),就拋出異常。
- NEVER:當(dāng)前的方法不應(yīng)該在事務(wù)中運(yùn)行,如果有運(yùn)行的事務(wù),就拋出異常。
- NESTED:如果有事務(wù)在運(yùn)行,當(dāng)前的方法就應(yīng)該在這個(gè)事務(wù)的嵌套事務(wù)內(nèi)運(yùn)行,否則,就啟動(dòng)一個(gè)新的事務(wù),并在自己的事務(wù)內(nèi)運(yùn)行。
1. REQUIRED
如果有事務(wù)在運(yùn)行,當(dāng)前方法就在這個(gè)事務(wù)內(nèi)運(yùn)行,否則就開啟一個(gè)新的事務(wù),并在自己的事務(wù)內(nèi)運(yùn)行。
@Transactional(propagation = Propagation.REQUIRED) public void methodA() { methodB(); // do something } @Transactional(propagation = Propagation.REQUIRED) public void methodB() { // do something }
當(dāng)調(diào)用 methodA 時(shí),因?yàn)楫?dāng)前上下文不存在事務(wù),所以會(huì)開啟一個(gè)新的事務(wù)。當(dāng)執(zhí)行到 methodB 時(shí),發(fā)現(xiàn)當(dāng)前上下文存在事務(wù),因此就加入到當(dāng)前事務(wù)來執(zhí)行。當(dāng)單獨(dú)調(diào)用 methodB 時(shí),因?yàn)楫?dāng)前上下文不存在事務(wù),所以會(huì)開啟一個(gè)新的事務(wù)。
2. REQUIRES_NEW
當(dāng)前的方法必須在啟動(dòng)新事務(wù),并在它自己的事務(wù)內(nèi)運(yùn)行,如果有事務(wù)在進(jìn)行,應(yīng)該將它掛起。
@Transactional(propagation = Propagation.REQUIRED) public void methodA() { doSomeThingA(); methodB(); doSomeThingB(); // do something else } // 事務(wù)屬性為REQUIRES_NEW @Transactional(propagation = Propagation.REQUIRES_NEW) public void methodB() { // do something }
當(dāng)調(diào)用 methodA 時(shí),開啟了 事務(wù) A。執(zhí)行到 methodB 時(shí),開啟一個(gè)事務(wù)B,此時(shí)事務(wù) A 掛起,當(dāng)事務(wù) B 執(zhí)行完成后,繼續(xù)執(zhí)行事務(wù) A 。
- 這里的事務(wù) A 稱為外層事務(wù)。
- 這里的事務(wù) B 則稱為內(nèi)層事務(wù)。
3. SUPPORTS
如果有事務(wù)在運(yùn)行,當(dāng)前的方法就在這個(gè)事務(wù)內(nèi)運(yùn)行,否則它可以不運(yùn)行在事務(wù)中。
@Transactional(propagation = Propagation.REQUIRED) public void methodA() { methodB(); // do something } // 事務(wù)屬性為SUPPORTS @Transactional(propagation = Propagation.SUPPORTS) public void methodB() { // do something }
當(dāng)調(diào)用 methodA 時(shí),methodB 加入到 methodA 的事務(wù)中,事務(wù)的執(zhí)行。當(dāng)單獨(dú)的調(diào)用 methodB 時(shí),methodB方法是非事務(wù)的執(zhí)行的。
4. NOT_SUPPORTED
當(dāng)前的方法不應(yīng)該運(yùn)行在事務(wù)中,如果有運(yùn)行的事務(wù),將它掛起。
- 當(dāng)調(diào)用 methodA 時(shí),開啟 事務(wù) A。當(dāng)執(zhí)行到 methodB 時(shí),掛起事務(wù)A,以非事務(wù)的方式執(zhí)行 methodB 。
- 當(dāng)單獨(dú)的調(diào)用 methodB 時(shí),methodB方法是非事務(wù)的執(zhí)行的。
5. MANDATORY
當(dāng)前的方法必須運(yùn)行在事務(wù)內(nèi)部,如果沒有正在運(yùn)行的事務(wù),就拋出異常。
@Transactional(propagation = Propagation.REQUIRED) public void methodA() { methodB(); // do something } // 事務(wù)屬性為MANDATORY @Transactional(propagation = Propagation.MANDATORY) public void methodB() { // do something }
當(dāng)調(diào)用 methodA 時(shí),methodB 則加入到 methodA 的事務(wù)中,事務(wù)地執(zhí)行。當(dāng)單獨(dú)調(diào)用 methodB 時(shí),因?yàn)楫?dāng)前沒有一個(gè)活動(dòng)的事務(wù),則會(huì)拋出異常。
6. NEVER
當(dāng)前的方法不應(yīng)該在事務(wù)中運(yùn)行,如果有運(yùn)行的事務(wù),就拋出異常。
7. NESTED
如果有事務(wù)在運(yùn)行,當(dāng)前的方法就應(yīng)該在這個(gè)事務(wù)的嵌套事務(wù)內(nèi)運(yùn)行,否則,就啟動(dòng)一個(gè)新的事務(wù),并在自己的事務(wù)內(nèi)運(yùn)行。
- 當(dāng)單獨(dú)執(zhí)行 methodB ,開啟事務(wù) B,執(zhí)行。
- 當(dāng)執(zhí)行 methodA 時(shí),開啟事務(wù) A,執(zhí)行到 methodB,開啟內(nèi)層事務(wù) B。
注意 當(dāng)執(zhí)行 methodA 時(shí),這里是一個(gè)嵌套事務(wù)。
如果外層事務(wù)失敗,則會(huì)回滾內(nèi)層事務(wù)所做的操作。但是內(nèi)層事務(wù)的失敗不會(huì)引起外層事務(wù)的回滾。
二、ioslation
設(shè)置事務(wù)隔離級別。
因?yàn)椴煌氖聞?wù)隔離級別會(huì)引起不同的問題,比如:臟讀、不可重復(fù)讀、幻讀。
相關(guān)的內(nèi)容在之前有過介紹,有興趣的可以自行跳轉(zhuǎn)過去:
【Mysql】數(shù)據(jù)庫事務(wù),臟讀、幻讀、不可重復(fù)讀
所以要解決,需要設(shè)置對應(yīng)的隔離級別:
- READ_UNCOMMITTED: 讀未提交
- READ_COMMITTED: 讀已提交
- REPEATABLE_READ: 可重復(fù)讀
- SERIALIZABLE: 串行化
三、timeout
設(shè)置超時(shí)時(shí)間。
事務(wù)需要在一定時(shí)間內(nèi)進(jìn)行提交,如果沒提交,就回滾。
默認(rèn)值是 -1 ,表示不超時(shí)。如果設(shè)置時(shí)間,單位是秒(s)。
四、readOnly
設(shè)置是否只讀。
默認(rèn)值 false,表示可以查詢,也可以進(jìn)行添加、修改、刪除操作。
當(dāng)設(shè)置為 true,只可以進(jìn)行查詢操作。
五、rollbackFor
設(shè)置出現(xiàn)哪些異常后就進(jìn)行事務(wù)的回滾。
六、noRollbackFor
設(shè)置出現(xiàn)哪些異常后,不進(jìn)行事務(wù)的回滾。
以上就是Spring學(xué)習(xí)JdbcTemplate數(shù)據(jù)庫事務(wù)參數(shù)的詳細(xì)內(nèi)容,更多關(guān)于Spring JdbcTemplate數(shù)據(jù)庫事務(wù)的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Springcloud GateWay網(wǎng)關(guān)配置過程圖解
這篇文章主要介紹了Springcloud GateWay網(wǎng)關(guān)配置過程圖解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-12-12spring boot使用@Async異步注解的實(shí)現(xiàn)原理+源碼
通常我們都是采用多線程的方式來實(shí)現(xiàn)上述業(yè)務(wù)功能,但spring 提供更優(yōu)雅的方式來實(shí)現(xiàn)上述功能,就是@Async 異步注解,在方法上添加@Async,spring就會(huì)借助AOP,異步執(zhí)行方法,接下來通過本文給大家介紹spring boot異步注解的相關(guān)知識,一起看看吧2021-06-06淺談利用Spring的AbstractRoutingDataSource解決多數(shù)據(jù)源的問題
本篇文章主要介紹了淺談利用Spring的AbstractRoutingDataSource解決多數(shù)據(jù)源的問題,具有一定的參考價(jià)值,有需要的可以了解一下2017-08-08關(guān)于Java中static關(guān)鍵字的用法
這篇文章主要介紹了關(guān)于Java中static關(guān)鍵字的用法,static:意為靜態(tài)的,在?Java?里面作為靜態(tài)修飾符,可以理解為全局的意思,static?不僅可以修飾成員變量,成員方法,還可以修飾代碼塊,需要的朋友可以參考下2023-08-08