java中事务@Transaction的正确使用和触发回滚机制【经典】

@Transactional(rollbackFor=Exception.class) 的作用

1.1 说明

|-------------------------------------------------------------------------------------------------|
| @Transactional(rollbackFor = Exception.class) public void updateStock(Product targetProduct) {} |

作用:

1.开启数据库事务(Transaction)

2.强制对所有异常进行回滚:明确告诉 Spring:"只要方法抛出任何 Exception(包括 IOException、自定义业务异常等),都必须回滚事务!"

二****触发事务回滚机制

2.1说明

Spring 的声明式事务(@Transactional)是通过AOP代理实现的。它的回滚机制依赖于方法执行过程中是否抛出异常并传递到代理层。

✅ 会回滚的情况:方法内部抛出异常,并且这个异常没有被捕获,或者被捕获后重新抛出,最终导致方法以异常形式结束。

❌ 不会回滚的情况:方法内部使用try-catch捕获了所有异常,并在catch块中吞掉异常(不抛出),使得方法正常结束(return)。

三 反例说明

3.1 反例吞并异常,无法回滚

即使 applyForCheckEdit 抛出了 SQLException 或其他 Exception,你的 catch 块只是记录日志、放一个错误码到 resultMap,然后让方法正常返回。

Spring 的事务代理看到的是:"这个方法成功执行完了,返回了一个 Map",根本不知道中间出过错。因此,事务会被提交(commit),而不是回滚!

后果:如果 applyForCheckEdit 执行了一半(比如更新了 A 表,但在更新 B 表时出错),由于事务被提交,会导致数据不一致。

结论:**这段代码中的 @Transactional(rollbackFor = Exception.class) 不会生效,事务不会回滚。**只要你在 catch 块里"吃掉"了异常,事务就永远不会回滚。

3.2 解决办法

3.2.1 方案1:不进行try catch捕获,原生抛出db异常,spring捕获触发回滚

如果你希望任何错误都导致整个批量操作回滚,就不要用 try-catch 包裹数据库操作:

3.2.2 方案2:捕获后重新抛出(保留自定义处理逻辑)

如果你确实需要在 catch 中记录日志或做清理,但又希望回滚,必须重新抛出异常:

注意:一旦你 throw 了,调用方就需要处理这个异常,不能再通过返回 Map 判断成功与否。

相关推荐
wand codemonkey14 分钟前
SpringbootWeb【入门】+MySQL【安装】+【DataDrip安装 】+【连接MySQL】
java·mysql·mybatis
Highcharts.js1 小时前
缺失数据可视化图表开发实战|Highcharts创建人员出生统计面积图表示例
开发语言·前端·javascript·信息可视化·highcharts·图表开发
测试员周周6 小时前
【Appium 系列】第16节-WebView-H5上下文切换 — 混合应用的自动化难点
运维·开发语言·人工智能·功能测试·appium·自动化·测试用例
Mahir088 小时前
Spring 循环依赖深度解密:从问题本质到三级缓存源码级解析
java·后端·spring·缓存·面试·循环依赖·三级缓存
杜子不疼.8 小时前
【C++ AI 大模型接入 SDK】 - DeepSeek 模型接入(上)
开发语言·c++·chatgpt
加号38 小时前
【C#】 串口通信技术深度解析及实现
开发语言·c#
sycmancia9 小时前
Qt——编辑交互功能的实现
开发语言·qt
RyFit9 小时前
SpringAI 常见问题及解决方案大全
java·ai
石山代码9 小时前
C++ 内存分区 堆区
java·开发语言·c++
绝知此事10 小时前
【算法突围 01】线性结构与哈希表:后端开发的收纳术
java·数据结构·算法·面试·jdk·散列表