日韩无码专区无码一级三级片|91人人爱网站中日韩无码电影|厨房大战丰满熟妇|AV高清无码在线免费观看|另类AV日韩少妇熟女|中文日本大黄一级黄色片|色情在线视频免费|亚洲成人特黄a片|黄片wwwav色图欧美|欧亚乱色一区二区三区

RELATEED CONSULTING
相關(guān)咨詢
選擇下列產(chǎn)品馬上在線溝通
服務(wù)時間:8:30-17:00
你可能遇到了下面的問題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營銷解決方案
盤點Spring事務(wù)失效的4種寫法及解決方案,Review代碼再也不慌了

1、非運行時異常導(dǎo)致事務(wù)無法回滾

我們知道,Spring是通過AOP的方式來實現(xiàn)事務(wù)的,而在處理事務(wù)的過程中,Spring只有捕獲到RuntimeException或者Error的時候才會觸發(fā)回滾操作,如果我們在代碼中拋出的是非運行時異常,而又沒有特殊配置的話,事務(wù)就會無法回滾。

成都創(chuàng)新互聯(lián)專注于萬州企業(yè)網(wǎng)站建設(shè),響應(yīng)式網(wǎng)站設(shè)計,商城系統(tǒng)網(wǎng)站開發(fā)。萬州網(wǎng)站建設(shè)公司,為萬州等地區(qū)提供建站服務(wù)。全流程按需定制網(wǎng)站,專業(yè)設(shè)計,全程項目跟蹤,成都創(chuàng)新互聯(lián)專業(yè)和態(tài)度為您提供的服務(wù)

下面我們以一個簡單的例子,復(fù)現(xiàn)一下這種情況,以及針對這種情況的解決方案。

本文Springboot版本:2.7.6,數(shù)據(jù)源為MySQL。

首先創(chuàng)建一個測試用的User對象:

@Data
public class User {
@TableId(type = IdType.AUTO)
private Integer id;
private String name;
private String pwd;
}

建表語句:

CREATE TABLE user  (
`id` int(11) NOT NULL AUTO_INCREMENT,
`name` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL,
`pwd` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL,
PRIMARY KEY (`id`) USING BTREE
) ENGINE = InnoDB;

測試邏輯:往user表插入一條數(shù)據(jù),如果插入成功,就拋出exception異常,測試數(shù)據(jù)是否回滾。

@Service
@AllArgsConstructor
public class DemoService {
private final UserMapper userMapper;

@Transactional
public ResponseEntity addUser(User user) throws Exception {
int insert = this.userMapper.insert(user);
if (insert > 0) {
throw new Exception("異?;貪L測試");
}
return ResponseEntity.ok().build();
}
}

新建測試方法:

@Test
public void addUserTest() throws Exception {
User user = new User();
user.setName("測試");
user.setPwd("123456");
this.demoService.addUser(user);
}

運行測試方法,從控制臺可以看到,我們手動指定的異常被成功拋出。

但是,當(dāng)異常發(fā)生時,事務(wù)并沒有被回滾,數(shù)據(jù)依然被插入到了數(shù)據(jù)庫。

解決辦法:
1,將異常包裝成運行時異常:throw new RuntimeException("異?;貪L測試");

2,在@Transactional指定回滾的異常類型,@Transactional(rollbackFor = Exception.class)。

一般來說,使用第二種方式會更清晰一些,但是有些朋友往往會忘記手動指定回滾的異常類型,進(jìn)而導(dǎo)致非預(yù)期的bug產(chǎn)生。

2、通過this調(diào)用本類事務(wù)方法導(dǎo)致的事務(wù)無法回滾

隨著業(yè)務(wù)的發(fā)展,核心業(yè)務(wù)代碼會越來越多,同一個方法也會越寫越長。我們?yōu)榱耸勾a邏輯更加高內(nèi)聚低耦合,會將功能相同的代碼進(jìn)行封裝成一個個的子方法。

但是,如果我們對事務(wù)的運行機制了解不透徹,隨意在同一個類中通過this調(diào)用事務(wù)方法,就可能導(dǎo)致非預(yù)期的bug。

@Service
@RequiredArgsConstructor
public class DemoService {
private final UserMapper userMapper;
public ResponseEntity addUser(User user){
//注意這一行
this.doAddUser(user);
return ResponseEntity.ok().build();
}

@Transactional(rollbackFor = Exception.class)
public void doAddUser(User user) {
int insert = this.userMapper.insert(user);
if (insert > 0) {
throw new RuntimeException("測試添加異?;貪L");
}
}
}

如以上代碼所示,在addUser方法中調(diào)用了事務(wù)方法doAddUser,如果數(shù)據(jù)插入成功,就拋出一個異常,測試數(shù)據(jù)是否能夠回滾。

通過測試用例可以看到,異常已經(jīng)拋出,但是數(shù)據(jù)庫中卻成功的插入了數(shù)據(jù),我們期望的數(shù)據(jù)并沒有回滾。

原因探究:

原因其實很簡單,通過this方法調(diào)用時,Spring的代理沒能起作用,事務(wù)自然也就無法介入,關(guān)于這一點的原理在之前的文章中也有分析過,感興趣的朋友可以去看一看。

有的朋友可能會說,項目的代碼已經(jīng)是這樣了,再將老方法重寫到新類中也不現(xiàn)實,有沒有辦法改動較小的方式呢?

其實很簡單,現(xiàn)在事務(wù)失效的原因是代理失效,那么想辦法讓代理重新生效就行了。

我們在本類中注入一個當(dāng)前對象,這個對象可以被Spring代理,那么這個對象的方法自然也可以被代理。

@Service
@RequiredArgsConstructor
public class DemoService {
private final UserMapper userMapper;
@Resource
private DemoService self;
public ResponseEntity addUser(User user){
//通過self引用使代理生效
this.self.doAddUser(user);
return ResponseEntity.ok().build();
}

@Transactional(rollbackFor = Exception.class)
public void doAddUser(User user) {
int insert = this.userMapper.insert(user);
if (insert > 0) {
throw new RuntimeException("測試添加異?;貪L");
}
}
}

3、被聲明的事務(wù)方法是private類型

這種錯誤在博主剛工作時遇到挺多次的,不過現(xiàn)在現(xiàn)代IDE已經(jīng)越來越智能了,對于這種情況會直接給出錯誤提示,所以這里提出這種錯誤只是告訴大家,事務(wù)方法是不能聲明為private的。

至于為什么不能是private,那自然還是和代理有關(guān)了。

4、嵌套事務(wù)異常導(dǎo)致事務(wù)被提前關(guān)閉而報錯

當(dāng)使用嵌套事務(wù)時,需要明確指定事務(wù)的傳播范圍。

@Service
@RequiredArgsConstructor
public class DemoService {
private final UserMapper userMapper;
@Resource
private DemoService self;

@Transactional(rollbackFor = Exception.class)
public ResponseEntity addUser(User user) {
int insert = this.userMapper.insert(user);
if (insert > 0) {
try {
this.self.update(user);
} catch (Exception e) {
System.out.println("即使更新異常也不要影響添加數(shù)據(jù)");
}
}
return ResponseEntity.ok().build();
}

@Transactional(rollbackFor = Exception.class)
public void update(User user) {
user.setPwd("666666");
int update = this.userMapper.updateById(user);
if (update > 0) {
throw new RuntimeException("測試更新數(shù)據(jù)回滾");
}
}
}

如以上代碼,我們添加完一條數(shù)據(jù)之后,嘗試將密碼更新為666666,并且希望即使更新異常,也不要影響添加操作。

然而運行測試用例,我們會得到這樣一條錯誤信息:
org.springframework.transaction.UnexpectedRollbackException: Transaction rolled back because it has been marked as rollback-only。

什么意思呢?就是當(dāng)Spring處理事務(wù)時,發(fā)現(xiàn)事務(wù)已經(jīng)被回滾了。

這是因為我們并沒有指定事務(wù)的傳播行為,默認(rèn)情況下,Spring的事務(wù)傳播是REQUIRED,即:如果本來有事務(wù),則加入該事務(wù),如果沒有事務(wù),則創(chuàng)建新的事務(wù)。

我們添加數(shù)據(jù)時啟動了一個事務(wù),更新數(shù)據(jù)時,Spring判斷當(dāng)前已經(jīng)存在事務(wù),所以就不再新建事務(wù),而是加入當(dāng)前事務(wù)。

但是當(dāng)更新操作失敗時,需要對事務(wù)進(jìn)行回滾,更新是沒問題的,正?;貪L。

但是插入操作就不行了,當(dāng)要提交插入操作的事務(wù)時,由于事務(wù)已經(jīng)被回滾了,無法再次操作,Spring只好報錯來提示我們了。

如何處理呢?在更新操作上指明事務(wù)的傳播范圍就行。

@Transactional(rollbackFor = Exception.class,propagation = Propagation.REQUIRES_NEW)
public void update(User user){
user.setPwd("666666");
int update = this.userMapper.updateById(user);
if (update > 0) {
throw new RuntimeException("測試更新數(shù)據(jù)回滾");
}
}

再測試一下,發(fā)現(xiàn)插入操作的事務(wù)可以正常提交了。

總結(jié)

事務(wù)是我們?nèi)粘i_發(fā)工作中無法避免的一個功能,深刻理解事務(wù)的運行機制,正確使用事務(wù)的聲明式操作,才能讓我們寫出更健壯的代碼。


當(dāng)前標(biāo)題:盤點Spring事務(wù)失效的4種寫法及解決方案,Review代碼再也不慌了
文章分享:http://www.5511xx.com/article/cdsoges.html