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將CSV的數(shù)據(jù)發(fā)送到kafka的示例
這篇文章主要介紹了Java將CSV的數(shù)據(jù)發(fā)送到kafka得示例,幫助大家更好得理解和使用Java,感興趣的朋友可以了解下2020-11-11
Spring中的@EnableConfigurationProperties使用方式以及作用詳解
這篇文章主要介紹了Spring中的@EnableConfigurationProperties使用方式以及作用詳解,使用了?@ConfigurationProperties?注解的配置類生效,將該類注入到?IOC?容器中,交由?IOC?容器進(jìn)行管理,此時則不用再配置類上加上@Component,需要的朋友可以參考下2024-01-01
Java實(shí)現(xiàn)學(xué)生信息管理界面
這篇文章主要為大家詳細(xì)介紹了Java實(shí)現(xiàn)學(xué)生信息管理界面,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-06-06

