解决try-catch捕获异常信息后Spring事务失效的问题
作者:DT辰白 时间:2022-11-15 03:17:33
一、首先在Spring Boot项目中,手动添加异常方法进行测试
@Transactional(rollbackFor=Exception.class) //表示此方法有异常时触发Spring事务
@Override
public CommonResult<User> saveUser(User user) {
int insert = baseMapper.insert(user);
try {
// 添加异常,并进行捕获
int a = 10/0;
}catch (Exception e){
logger.info("打印异常信息:"+e);
return CommonResult.commentFailure("服务器异常,事务回滚");
}
if(insert > 0){
return CommonResult.commentSuccess(user);
}else {
return CommonResult.commentFailure("添加失败");
}
}
1、一个添加信息的实现类方法上,此处我们加了Spring的事务。
2、问题:一个方法报异常(int a = 10/0)进行了异常捕获,另一个方法不会回滚(insert添加方法)
这是什么情况呢,相当于Spring事务策略失效了。
try-catch捕获了异常后,这种业务方法也就等于脱离了spring事务的管理,因为没有任何异常会从业务方法中抛出,全被捕获并“吞掉”,导致spring异常抛出触发事务回滚策略失效。
通俗的来说:默认spring事务只在发生未被捕获的 runtimeexcetpion或error时才回滚。
二、处理方案一
spring aop 异常捕获进而回滚。在catch中最后加上throw new runtimeexcetpion(),这样程序异常时才能被aop捕获进而回滚,缺点是无法return异常信息提示,前端用户交互效果不佳
@Transactional(rollbackFor=Exception.class) //表示此方法有异常时触发Spring事务
@Override
public CommonResult<User> saveUser(User user) {
int insert = baseMapper.insert(user);
try {
// 添加异常,并进行捕获
int a = 10/0;
}catch (Exception e){
logger.info("异常信息:"+e);
// 方案一:spring aop 异常捕获
throw new RuntimeException();
}
if(insert > 0){
return CommonResult.commentSuccess(user);
}else {
return CommonResult.commentFailure("添加失败");
}
}
三、处理方案二
就是让一个方法报异常,另一个方法回滚,这样才能真正的触发Spring事务回滚策略。
catch语句中增加:
TransactionAspectSupport.currentTransactionStatus().setRollbackOnly(); //手动回滚,这样上层就无需去处理异常了
完整代码:
@Transactional(rollbackFor=Exception.class) //表示此方法有异常时触发Spring事务
@Override
public CommonResult<User> saveUser(User user) {
int insert = baseMapper.insert(user);
try {
// 添加异常,并进行捕获
int a = 10/0;
}catch (Exception e){
logger.info("异常信息:"+e);
// 方案二:手动回滚
TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
return CommonResult.commentFailure("服务器异常,事务回滚");
}
if(insert > 0){
return CommonResult.commentSuccess(user);
}else {
return CommonResult.commentFailure("添加失败");
}
}
四、如过需要手动进行手动回滚的业务方法比较多,我们可以写一个公共的工具类
SpringRollBackUtil.java
public class SpringRollBackUtil {
/**
* 事务回滚机制
*/
public static void rollBack() {
try {
TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
} catch (Exception e) {
e.printStackTrace();
}
}
}
只需调用方法即可
// 方案三:公共工具类 手动回滚
SpringRollBackUtil.rollBack();
Spring mvc:事务引起的try/catch失效
在测试一个接口时,发现一个奇怪的现象:
该接口使用@ResponseBody注解返回json格式数据,并且使用try/catch包括全部逻辑代码,debug后发现返回数据没有任何错误,只包含一段因产生异常导致的错误提示字符串,但是chrome浏览器network却显示http状态码为500。
最后发现在该RequestMapping方法上还有一个注解@Transactional,去除ok。
来源:https://blog.csdn.net/qq_41107231/article/details/106698940
标签:try-catch,异常,Spring事务
![](/images/zang.png)
![](/images/jiucuo.png)
猜你喜欢
详解C#中 Thread,Task,Async/Await,IAsyncResult的那些事儿
2021-12-08 20:07:32
![](https://img.aspxhome.com/file/2023/4/129684_0s.png)
Java中的notyfy()和notifyAll()的本质区别
2022-06-05 22:46:19
idea无法切换分支报错问题及解决
2023-12-14 22:39:26
![](https://img.aspxhome.com/file/2023/2/64552_0s.png)
Java synchronized轻量级锁实现过程浅析
2022-05-08 07:28:55
![](https://img.aspxhome.com/file/2023/6/84606_0s.png)
MyBatis动态SQL如何实现前端指定返回字段
2023-11-28 23:00:58
如何用.NETCore操作RabbitMQ
2022-06-20 04:11:04
![](https://img.aspxhome.com/file/2023/8/76378_0s.png)
Spring security如何重写Filter实现json登录
2023-09-15 13:33:31
GC调优实战之高分配速率High Allocation Rate
2021-09-30 22:03:21
android tv列表焦点记忆实现的方法
2023-08-25 19:36:53
C++实现的求解多元一次方程示例
2022-09-26 19:45:47
Java聊天室之使用Socket实现通信功能
2022-03-08 09:46:27
![](https://img.aspxhome.com/file/2023/7/69117_0s.png)
基于jni调用时,jvm报错问题的深入分析
2022-08-08 17:21:55
C#调用JS的几种方法
2022-09-29 23:24:18
java冷知识:javac AbstractProcessor详解
2022-08-01 19:32:09
![](https://img.aspxhome.com/file/2023/6/66626_0s.jpg)
c语言定时器示例分享
2023-11-04 09:44:52
Android编程实现简单文件浏览器功能
2022-01-02 06:13:46
![](https://img.aspxhome.com/file/2023/8/99798_0s.jpg)
java多种幻灯片切换特效(经典)
2022-08-05 23:18:22
![](https://img.aspxhome.com/file/2023/5/95935_0s.jpg)
Java通过Scanner了解if...else if语句
2023-11-29 04:28:44
使用itextpdf解决PDF合并的问题
2023-09-21 04:47:16
.NET WinForm实现在listview中添加progressbar的方法
2021-10-08 20:23:54