完美解决Spring声明式事务不回滚的问题
发布时间 - 2026-01-11 01:38:14 点击率:次疑问,确实像往常一样在service上添加了注解 @Transactional,为什么查询数据库时还是发现有数据不一致的情况,想想肯定是事务没起作用,出现异常的时候数据没有回滚。于是就对相关代码进行了一番测试,结果发现一下踩进了两个坑,确实是事务未回滚导致的数据不一致。

下面总结一下经验教训:
Spring事务的管理操作方法
编程式的事务管理
实际应用中很少使用
通过使用TransactionTemplate 手动管理事务
声明式的事务管理
开发中推荐使用(代码侵入最少)
Spring的声明式事务是通过AOP实现的
主要掌握声明式的事务管理。
spring事务不回滚的两个原因
总结一下导致事务不回滚的两个原因,一是Service类内部方法调用,二是try...catch异常。
1. Service类内部方法调用
大概就是 Service 中有一个方法 A,会内部调用方法 B, 方法 A 没有事务管理,方法 B 采用了声明式事务,通过在方法上声明 Transactional 的注解来做事务管理。示例代码如下:
@Service
public class RabbitServiceImpl implements RabbitService {
@Autowired
private RabbitDao rabbitDao;
@Autowired
private TortoiseDao tortoiseDao;
@Override
public Rabbit methodA(String name){
return methodB(name);
}
@Transactional(propagation = Propagation.REQUIRED)
public boolean methodB(String name){
rabbitDao.insertRabbit(name);
tortoiseDao.insertTortoise(name);
return true;
}
}
单元测试代码如下:
public class RabbitServiceImplTest {
@Autowired
private RabbitService rabbitService;
// 事务未开启
@Test
public void testA(){
rabbitService.methodA("rabbit");
}
// 事务开启
@Test
public void testB(){
rabbitService.methodB("rabbit");
}
}
从上一节中可以看到,声明式事务是通通过AOP动态代理实现的,这样会产生一个代理类来做事务管理,而目标类(service)本身是不能感知代理类的存在的。
对于加了@Transactional注解的方法来说,在调用代理类的方法时,会先通过拦截器TransactionInterceptor开启事务,然后在调用目标类的方法,最后在调用结束后,TransactionInterceptor 会提交或回滚事务,大致流程如下图:
总结,在方法 A 中调用方法 B,实际上是通过“this”的引用,也就是直接调用了目标类的方法,而非通过 Spring 上下文获得的代理类,所以事务是不会开启的。
2. try...catch异常
在一段业务逻辑中对数据库异常进行了处理,使用了try...catch子句捕获异常并throw了一个自定义异常,这种情况导致了事务未回滚,示例代码如下:
@Transactional(propagation = Propagation.REQUIRED)
public boolean methodB(String name) throws BizException {
try {
rabbitDao.insertRabbit(name);
tortoiseDao.insertTortoise(name);
} catch (Exception e) {
throw new BizException(ReturnCode.EXCEPTION.code, ReturnCode.EXCEPTION.msg);
}
return true;
}
BizException的定义如下:
public class BizException extends Exception {
// 自定义异常
}
上面代码中的声明式事务在出现异常的时候,事务是不会回滚的。在代码中我虽然捕获了异常,但是同时我也抛出了异常,为什么事务未回滚呢?猜测是异常类型不对,于是开始查询原因,翻看了Spring的官方文档,找到了答案。下面是翻译自Spring官网。
17.5.3 声明式事务的回滚
上一节中介绍了如何设置开启Spring事务,一般在你的应用的Service层代码中设置,这一节将介绍在简单流行的声明式事务中如何控制事务回滚。
在Spring FrameWork 的事务框架中推荐的事务回滚方法是,在当前执行的事务上下文中抛出一个异常。如果异常未被处理,当抛出异常调用堆栈的时候,Spring FrameWork 的事务框架代码将捕获任何未处理的异常,然后并决定是否将此事务标记为回滚。
在默认配置中,Spring FrameWork 的事务框架代码只会将出现runtime, unchecked 异常的事务标记为回滚;也就是说事务中抛出的异常时RuntimeException或者是其子类,这样事务才会回滚(默认情况下Error也会导致事务回滚)。在默认配置的情况下,所有的 checked 异常都不会引起事务回滚。
注:Unchecked Exception包括Error与RuntimeException. RuntimeException的所有子类也都属于此类。另一类就是checked Exception。
你可以精确的配置异常类型,指定此异常类事务回滚,包括 checked 异常。下面的xml代码片段展示了如何配置checked异常引起事务回滚,应用自定义异常类型:
<tx:advice id="txAdvice" transaction-manager="txManager"> <tx:attributes> <tx:method name="get*" read-only="true" rollback-for="Exception"/> <tx:method name="*"/> </tx:attributes> </tx:advice>
与其有同等作用的注解形式如下:
@Transactional(rollbackForClassName={"Exception"})
或者
@Transactional(rollbackFor={Exception.class})
在你遇到异常不想回滚事务的时候,同样的你也可指定不回滚的规则,下面的一个例子告诉你,即使遇到未处理的 InstrumentNotFoundException 异常时,Spring FrameWork 的事务框架同样会提交事务,而不回滚。
<tx:advice id="txAdvice"> <tx:attributes> <tx:method name="updateStock" no-rollback-for="InstrumentNotFoundException"/> <tx:method name="*"/> </tx:attributes> </tx:advice>
与其有同样作用的注解形式如下:
@Transactional(noRollbackForClassName={"InstrumentNotFoundException"})
或者
@Transactional(noRollbackFor={InstrumentNotFoundException.class})
还有更灵活的回滚规则配置方法,同时指定什么异常回滚,什么异常不回滚。当Spring FrameWork 的事务框架捕获到一个异常的时候,会去匹配配置的回滚规则来决定是否标记回滚事务,使用匹配度最强的规则结果。因此,下面的配置例子表达的意思是,除了异常 InstrumentNotFoundException 之外的任何异常都会导致事务回滚。
<tx:advice id="txAdvice"> <tx:attributes> <tx:method name="*" rollback-for="Throwable" no-rollback-for="InstrumentNotFoundException"/> </tx:attributes> </tx:advice>
你也可以通过编程式的方式回滚一个事务,尽管方法非常简单,但是也有非常强的代码侵入性,使你的业务代码和Spring FrameWork 的事务框架代码紧密的绑定在一起,示例代码如下:
public void resolvePosition() {
try {
// some business logic...
} catch (NoProductInStockException ex) {
// trigger rollback programmatically
TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
}
}
如果可能的话,强烈推荐您使用声明式事务方式回滚事务,对于编程式事务,如果你强烈需要它,也是可以使用的,but its usage flies in the face of achieving a clean POJO-based architecture.(没懂...)
看完官方文档这节内容找到了问题的答案,原来是因为我们自定义的异常不是 RuntimeException。我的解决办法是,在注解@Transactional中添加 rollbackFor={BizException.class}。可能你会问我为什么不将自定义异常修改为继承RuntimeException,因为我需要BizException是一个checked 异常。
以上这篇完美解决Spring声明式事务不回滚的问题就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持。
# spring声明式事务回滚
# Spring声明式事务和@Aspect的拦截顺序问题的解决
# Spring的编程式事务和声明式事务详解
# SpringBoot 注解事务声明式事务的方式
# SpringMVC+MyBatis声明式事务管理
# Spring实现声明式事务的方法详解
# 不回
# 自定义
# 抛出
# 子类
# 给大家
# 来做
# 进行了
# 是一个
# 情况下
# 找到了
# 我也
# 如果你
# 文档
# 也有
# 是因为
# 子句
# 未处理
# 看了
# 也会
# 你可以
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
如何在万网ECS上快速搭建专属网站?
Laravel Seeder填充数据教程_Laravel模型工厂Factory使用
Laravel路由怎么定义_Laravel核心路由系统完全入门指南
实例解析Array和String方法
Linux网络带宽限制_tc配置实践解析【教程】
香港服务器如何优化才能显著提升网站加载速度?
Python文件流缓冲机制_IO性能解析【教程】
Android中Textview和图片同行显示(文字超出用省略号,图片自动靠右边)
Laravel如何实现密码重置功能_Laravel密码找回与重置流程
谷歌浏览器下载文件时中断怎么办 Google Chrome下载管理修复
如何在建站之星绑定自定义域名?
PHP怎么接收前端传的文件路径_处理文件路径参数接收方法【汇总】
Laravel如何记录自定义日志?(Log频道配置)
详解MySQL数据库的安装与密码配置
Claude怎样写结构化提示词_Claude结构化提示词写法【教程】
动图在线制作网站有哪些,滑动动图图集怎么做?
如何自定义建站之星网站的导航菜单样式?
Laravel全局作用域是什么_Laravel Eloquent Global Scopes应用指南
如何在IIS中新建站点并解决端口绑定冲突?
如何在橙子建站中快速调整背景颜色?
简单实现jsp分页
如何快速使用云服务器搭建个人网站?
如何在阿里云服务器自主搭建网站?
使用Dockerfile构建java web环境
Python企业级消息系统教程_KafkaRabbitMQ高并发应用
如何在IIS管理器中快速创建并配置网站?
JavaScript 输出显示内容(document.write、alert、innerHTML、console.log)
制作无缝贴图网站有哪些,3dmax无缝贴图怎么调?
Laravel怎么定时执行任务_Laravel任务调度器Schedule配置与Cron设置【教程】
Midjourney怎么调整光影效果_Midjourney光影调整方法【指南】
百度浏览器网页无法复制文字怎么办 百度浏览器复制修复
深入理解Android中的xmlns:tools属性
javascript日期怎么处理_如何格式化输出
如何在IIS中配置站点IP、端口及主机头?
Laravel Octane如何提升性能_使用Laravel Octane加速你的应用
国美网站制作流程,国美电器蒸汽鍋怎么用官方网站?
Laravel如何实现一对一模型关联?(Eloquent示例)
制作企业网站建设方案,怎样建设一个公司网站?
微博html5版本怎么弄发超话_超话进入入口及发帖格式要求【教程】
如何用AI一键生成爆款短视频文案?小红书AI文案写作指令【教程】
Laravel项目如何进行性能优化_Laravel应用性能分析与优化技巧大全
中山网站制作网页,中山新生登记系统登记流程?
北京企业网站设计制作公司,北京铁路集团官方网站?
laravel怎么用DB facade执行原生SQL查询_laravel DB facade原生SQL执行方法
如何彻底卸载建站之星软件?
头像制作网站在线观看,除了站酷,还有哪些比较好的设计网站?
Laravel怎么调用外部API_Laravel Http Client客户端使用
Laravel如何安装Breeze扩展包_Laravel用户注册登录功能快速实现【流程】
英语简历制作免费网站推荐,如何将简历翻译成英文?
Laravel如何与Pusher实现实时通信?(WebSocket示例)
上一篇:《同步学》加入班级方法
上一篇:《同步学》加入班级方法

