Spring事務(wù)傳播中嵌套調(diào)用實(shí)現(xiàn)方法詳細(xì)介紹
前言
最近在使用Spring框架時遇到了一些問題,主要是Spring的事務(wù)傳播問題,一個不帶事務(wù)的方法調(diào)用帶事務(wù)的方法,有時候會出現(xiàn)不回滾的情況,所以寫了這篇文章來記錄一下。
7種傳播方式
我們先來看Spring事務(wù)的7中傳播方式以及對應(yīng)的描述
屬性名稱 | 值 | 描述 |
---|---|---|
PROPAGATION__REQUIRED | REQUIRED | 表示的是當(dāng)前這個方法必須運(yùn)行在一個事務(wù)環(huán)境中,如果當(dāng)前方法已經(jīng)處于事務(wù)環(huán)境中,就可以直接使用該方法,否則開啟一個新的事務(wù) |
PROPAGATION_SUPPORTS | SUPPORTS | 如果當(dāng)前方法處于事務(wù)環(huán)境中,就使用當(dāng)前事務(wù),否則不使用事務(wù) |
PROPAGATION_MANDATORY | MANDATORY | 表示當(dāng)前方法一定要處于事務(wù)環(huán)境中,否則就拋出異常 |
PROPAGATION_REQUIRES_NEW | REQUIRES_NEW | 當(dāng)前方法需要運(yùn)行在新的事務(wù)中。如果當(dāng)前方法已在事務(wù)環(huán)境中,先暫停當(dāng)前事務(wù),在啟動新的事務(wù)方法后才執(zhí)行該方法,如果當(dāng)前方法不在事務(wù)環(huán)境中,就啟動一個新的事務(wù)后啟動執(zhí)行該方法。 |
PROPAGATION_NOT_SUPPORTED | NOT_SUPPORTED | 不支持當(dāng)前的事務(wù),總是以非事務(wù)狀態(tài)執(zhí)行。如果這個方法是事務(wù)方法,就先掛起這個事務(wù)方法,再執(zhí)行這個方法 |
PROPAGATION_NEVER | NEVER | 不支持當(dāng)前事務(wù),如果是事務(wù)方法,則拋出異常 |
PROPAGATION__NESTED | NESTED | 如果當(dāng)前執(zhí)行的方法處于事務(wù)環(huán)境中,依舊會啟動一個事務(wù),嵌套的事務(wù)也可以獨(dú)立于當(dāng)前事務(wù)獨(dú)立回滾和提交,如果當(dāng)前執(zhí)行的方法不在事務(wù)環(huán)境中,也會啟動一個新事務(wù)。 |
注解式事務(wù)
在Spring中,我們常用@Transactional來標(biāo)注一個事務(wù)方法,如果有點(diǎn)進(jìn)去這個注解的源碼都可以看到Spring對于添加這個注解的方法,都會默認(rèn)將這個方法的事務(wù)的傳播等級設(shè)置為REQUIRED,也就是是當(dāng)前方法必須處于一個事務(wù)方法中,或者使用調(diào)用這個方法的事務(wù)行為。
下面我們來分析下這個注解在什么情況下會失效,并且需要怎么樣來去避免這種情況的發(fā)生。
事務(wù)的方法之間的調(diào)用
下面這個例子模仿的是一個帶事務(wù)的方法調(diào)用另外一個事務(wù)方法,在下面的這個方法報(bào)錯,查看當(dāng)前事務(wù)有沒有進(jìn)行回滾
@Override @Transactional(rollbackFor = Exception.class) public void saveWithDish(SetmealDto setmealDto){ this.save(setmealDto); List<SetmealDish> setmealDishes = setmealDto.getSetmealDishes(); } @Transactional(rollbackFor = Exception.class) public void saveBa(List<SetmealDish> setmealDishes, SetmealDto setmealDto) throws Exception{ setmealDishes = setmealDishes.stream().peek((item) -> item.setSetmealId(setmealDto.getId())).collect(Collectors.toList()); setmealDishService.saveBatch(setmealDishes); int j = 2/0; }
根據(jù)事務(wù)的傳播等級來看,這種情況是肯定可以回滾的,但是如果是同一類中,像下面這種情況,同一個類中一個不帶事務(wù)的方法調(diào)用另外一個帶事務(wù)的方法,這種情況下它的事務(wù)會不會回滾呢?理論上我們覺得是會的,但是在測試的時候呢,我們發(fā)現(xiàn)這個事務(wù)并沒有進(jìn)行回滾,也就是說,這個事務(wù)注解@Transantional沒有生效
@Override public void saveWithDish(SetmealDto setmealDto) throws Exception{ List<SetmealDish> setmealDishes = setmealDto.getSetmealDishes(); saveBa(setmealDishes, setmealDto); } @Transactional(rollbackFor = Exception.class) public void saveBa(List<SetmealDish> setmealDishes, SetmealDto setmealDto) throws Exception{ this.save(setmealDto); setmealDishes = setmealDishes.stream().peek((item) -> item.setSetmealId(setmealDto.getId())).collect(Collectors.toList()); setmealDishService.saveBatch(setmealDishes); int j = 2/0; }
雖然這里我們報(bào)錯了,但是數(shù)據(jù)庫中還是新增了一條剛剛我們添加的一條數(shù)據(jù),這樣可以說明,這是沒有添加事務(wù)的,也驗(yàn)證了上面我們的方法。
下面我們來看情況上,當(dāng)不同類之間類方法的調(diào)用,如果一個事務(wù)方法調(diào)用一個非事務(wù)方法,這樣非事務(wù)方法當(dāng)然可以獲取到當(dāng)前這個事務(wù)的,不會開啟一個新的事務(wù)。但是當(dāng)一個非事務(wù)方法調(diào)用一個不同類的事務(wù)方法時,這樣會不會回滾呢,答案是會的,這邊我已經(jīng)進(jìn)行驗(yàn)證過了。
注意事項(xiàng)
我們需要記住Spring的默認(rèn)事務(wù)傳播等級是Required,在Spring掃描Bean時,會掃描這個方法是否帶有@Transactional注解,如果是包含的話,Spring會動態(tài)生成一個代理類(proxy),當(dāng)這個方法被調(diào)用時,是由代理類來進(jìn)行調(diào)用的,而在初始化時,同一個類下面,這個方法如果是沒有帶@Transactional注解調(diào)用一個@Transactional的方法的話,這個方法的調(diào)用是沒有經(jīng)過代理類的,就不會啟動transactional,也就是在同一個類出現(xiàn)無效的現(xiàn)象出現(xiàn)
所以,解決的話,我們可以將這兩個方法分開到兩個不同的類中,所以我們可以知道在一個service類中,如果一個非事務(wù)方法調(diào)用一個帶事務(wù)的方法和事務(wù)方法之間的相互調(diào)用都不會開啟新的事務(wù)。
到此這篇關(guān)于Spring事務(wù)傳播中嵌套調(diào)用實(shí)現(xiàn)方法詳細(xì)介紹的文章就介紹到這了,更多相關(guān)Spring嵌套調(diào)用內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java jwt使用公鑰字符串驗(yàn)證解析token鎖方法詳解
關(guān)于java獲取Token驗(yàn)證的問題相信很多人都遇見過,尤其是對剛接觸微信開發(fā)的人來說確實(shí)有點(diǎn)棘手,下面這篇文章主要給大家介紹了關(guān)于Java中token驗(yàn)證解析的相關(guān)資料,需要的朋友可以參考下2023-02-02SpringApplicationRunListener監(jiān)聽器源碼詳解
這篇文章主要介紹了SpringApplicationRunListener監(jiān)聽器源碼詳解,springboot提供了兩個類SpringApplicationRunListeners、SpringApplicationRunListener(EventPublishingRunListener),spring框架還提供了一個ApplicationListener接口,需要的朋友可以參考下2023-11-11Mybatis實(shí)現(xiàn)一對一、一對多關(guān)聯(lián)查詢的方法(示例詳解)
這篇文章主要介紹了Mybatis實(shí)現(xiàn)一對一、一對多關(guān)聯(lián)查詢的方法,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-04-04JavaSE實(shí)現(xiàn)圖書管理系統(tǒng)的示例代碼
這篇博客是在學(xué)習(xí)了一部分Java基礎(chǔ)語法之后的練習(xí)項(xiàng)目,通過這個小項(xiàng)目的練習(xí),對Java中的類和對象,抽象類和接口等進(jìn)行熟悉理解??旄S小編一起學(xué)習(xí)學(xué)習(xí)吧2022-08-08對Java ArrayList的自動擴(kuò)容機(jī)制示例講解
今天小編就為大家分享一篇對Java ArrayList的自動擴(kuò)容機(jī)制示例講解,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-10-10