当前位置: 首页 > news >正文

《JMS事务性会话彻底解析:消息监听中的 commit、rollback 和幂等设计》

大家好,我是G探险者!

📌 场景引入

在实际项目中,我们常常面临以下挑战:

  • 监听 MQ 消息失败了,希望自动重试?
  • 消费 MQ 消息后,要写数据库,但中间报错了?
  • 消息处理必须要么成功要么失败,否则可能导致脏数据?
  • 消息是幂等的吗?可以重复投递处理吗?

这些都需要 事务性会话 + 容器回滚机制 + 幂等控制 组合拳来解决。


✅ 一、什么是 JMS 的事务性会话?

事务性会话(transacted = true)是一种 将消息的接收与处理放入事务中控制 的机制。

与确认模式(acknowledge)对比:

特性确认模式(ACK)事务性会话(Transacted)
消息确认AUTO_ACKNOWLEDGECLIENT_ACKNOWLEDGE 等使用 session.commit()
回滚方式手动控制 ACK抛异常或手动 session.rollback()
MQ是否重发消息否(默认不重发)✅ 是,失败自动重新投递
一次事务包含消息数一条(Spring容器下)✅ 默认一条,支持手动多条

🛠️ 二、Spring 如何开启事务性监听?

Spring 中的 DefaultMessageListenerContainer 支持事务模式:

DefaultMessageListenerContainer container = new DefaultMessageListenerContainer();
container.setConnectionFactory(connectionFactory);
container.setDestinationName("MY.QUEUE");
container.setMessageListener(new MyListener());container.setSessionTransacted(true); // ✅ 开启事务会话
container.setSessionAcknowledgeMode(Session.SESSION_TRANSACTED); // 推荐container.afterPropertiesSet();
container.start();

这样配置后,每条消息的处理会包裹在如下事务中:

Session session = connection.createSession(true, Session.SESSION_TRANSACTED)

🧪 三、事务处理机制详解

Spring 容器负责自动控制事务行为:

try {messageListener.onMessage(message);session.commit(); // ✅ 成功后提交
} catch (Throwable ex) {session.rollback(); // ❌ 抛异常后回滚,MQ 重发消息throw ex;
}

✅ 你只要记住:

  • 成功就正常返回(容器帮你 commit)
  • 失败就抛出异常(容器自动 rollback)

🔂 四、消息重试机制联动

Spring rollback → MQ 检测未 commit → 触发重投

🔁 每个 MQ 中间件(IBM MQ、ActiveMQ、TongLinkQ 等)都支持配置:

  • 最大重投次数
  • 重投间隔(redelivery delay)
  • 超过重试后投递到死信队列(DLQ)

💥 五、事务作用范围:是“一条消息”吗?

这个问题很关键,我们以 Spring 默认配置为例说明:

场景事务作用范围
DefaultMessageListenerContainer 默认行为✅ 每条消息单独包裹事务
自定义 Session 拉多条消息后统一 commit❌ 多条消息为一个事务
设置并发消费者(线程池)每条消息独立事务(线程隔离)

实战建议:

✅ 在监听容器中消费 MQ 消息,默认一条消息就是一个事务单元,安全可靠。


🎯 六、事务 + 幂等的设计建议

事务只能解决“要么成功、要么失败”的问题,不能避免重复处理

所以业务系统通常要配合幂等性策略:

场景幂等性设计建议
写数据库利用主键/唯一索引避免重复插入
写 Redis使用 SETNX 保证消息只处理一次
写业务日志使用消息 ID 做去重处理
第三方调用如果不能重复调用,要做幂等屏障

☑️ 七、监听失败常见问题排查

问题排查建议
没开启事务?是否调用了 setSessionTransacted(true)
容器未启动?是否漏了 afterPropertiesSet() 调用
消息处理失败后 MQ 不重发?是否吞掉异常了?应抛出异常给容器
重投失败消息去哪了?查看 MQ 的 DLQ(死信队列)配置

📘 小结

功能点建议配置
自动控制 commit/rollback使用 DefaultMessageListenerContainer
每条消息开启事务setSessionTransacted(true)
抛异常触发回滚在 onMessage() 中不要吞异常
幂等设计配合事务做幂等逻辑
消息处理失败自动重试借助 MQ 的重投策略 + Spring 回滚机制

📘 下一篇预告:

第 4 篇:《JMS 消息重试机制与死信队列配置指南》
我们将详细讲解 MQ 的 redelivery policy、最大重试次数配置、死信队列处理策略,以及如何在 Spring 中优雅应对消费失败。


http://www.dtcms.com/a/263289.html

相关文章:

  • 每天一个前端小知识 Day 17 - 微前端架构实战与 Module Federation
  • 记录H5内嵌到flutter App的一个问题,引发后面使用fastClick,引发后面input输入框单击无效问题。。。
  • BI软件选型:7款可私有部署产品对比
  • 利用不坑盒子的Copilot,快速排值班表
  • 在 Vue3 + Element Plus 中实现 el-table 拖拽排序功能
  • 【c语言课程设计】单选题考试系统(无链表,含码源)
  • 多校区在线跑腿小程序源码系统搭建平台 PHP+MySQL组合开发 含完整的搭建教程
  • 商品中心—16.库存分桶调配的技术文档
  • 【分布式】自定义统一状态机流转设计
  • Flowable01SpringBoot项目的引入--------------------每天都会更新,自学中
  • 组成原理精讲课--硬布线控制器和微程序控制器
  • STM32之火焰传感器模块(四针)
  • 11、类加载器
  • 项目:数据库应用系统开发:智能电商管理系统
  • 【Springai】项目实战进度和规划
  • 【FR801xH】富芮坤FR801xH之PMU GPIO
  • OpenCV CUDA模块设备层----- 正切(tangent)运算函数tan()
  • Python 数据分析与机器学习入门 (五):Matplotlib 数据可视化基础
  • R1-Searcher使用强化学习增强语言模型解决问题的搜索能力
  • WebSocket 的核心原理和工作流程
  • 前端Vue面试八股常考题(一)
  • 企业流程知识:《超越再造:以流程为中心的组织如何改变我们的工作和生活》读书笔记
  • 力扣面试150(7/150)
  • 【c/c++2】多线程,动静态库,信号,socket
  • 如何让宿主机完全看不到Wi-Fi?虚拟机独立联网隐匿上网实战!
  • 【知识图谱构建系列7】:结果评价(1)
  • 可编辑39页PPT | 数字化工厂蓝图整体框架建设举措建设路径实施路线规划建设方案
  • 从入门到精通:npm、npx、nvm 包管理工具详解及常用命令
  • Microsoft Edge 打开无反应、打开后显示兼容性问题、卸载重装 解决方案。一键卸载Microsoft Edge 。
  • 卫朋:华为流程体系拆解系列——IPD流程L1-L6分级导入实战演练