《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.pswp.cn/bicheng/87020.shtml
繁体地址,请注明出处:http://hk.pswp.cn/bicheng/87020.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

vue3 el-table 列增加 自定义排序逻辑

在 Vue 3 中使用 Element Plus 的 <el-table> 组件时&#xff0c;如果你想增加自定义排序逻辑&#xff0c;可以通过以下几个步骤实现&#xff1a; 1. 使用 default-sort 属性 首先&#xff0c;你可以在 <el-table> 组件上使用 default-sort 属性来指定默认的排序…

ISP Pipeline(7): Gamma Correction 伽马校正

AI_Plays/ISP/Fast_ISP_Progress.ipynb at main ameengee/AI_Plays GitHub Gamma Correction&#xff08;伽马校正&#xff09;是图像处理中的一个重要步骤&#xff0c;目的是调整图像的亮度&#xff0c;使其更符合人眼的感知或显示设备的特性。 为什么需要 Gamma Correcti…

AI提取伴奏,实现卡拉OK效果 —— 「suno api/luno api/kuka api」

导读 喜欢唱歌&#xff0c;却总苦于找不到纯净的伴奏&#xff1f;或者你想把喜欢的歌曲翻唱一遍&#xff0c;却被人声干扰搞得头大&#xff1f;现在&#xff0c;AI技术已经悄悄解决了这个问题。借助AI智能工具&#xff0c;你可以轻松提取任何一首歌的伴奏&#xff0c;享受宛如…

pip介绍

pip是什么&#xff1f; pip&#xff08;Pip Installs Packages&#xff09;是Python的官方管理工具&#xff0c;用于安装、升级、卸载和管理Python第三方库及其依赖关系。它是Python生态系统的核心组件&#xff0c;通过连接PyPI&#xff08;Python Package Index&#xff09;这…

机器学习20-线性网络思考

机器学习20-线性网络思考 针对线性网络的基础问题&#xff0c;使用基础示例进行解释 1-核心知识点 1-线性模型家族的线性回归和逻辑回归分别是什么&#xff0c;线性模型家族还有没有其他的模型 线性模型家族是一系列基于线性假设的统计模型&#xff0c;它们假设因变量和自变量…

【科研绘图系列】R语言绘制世界地图分布(world map)

禁止商业或二改转载,仅供自学使用,侵权必究,如需截取部分内容请后台联系作者! 文章目录 介绍加载R包数据下载导入数据数据预处理准备画图画图总结系统信息介绍 本教程旨在通过R语言及其相关地理空间分析包,展示如何对环境数据进行空间聚类分析,并将结果可视化。教程从读…

Armbian 25.5.1 Noble Gnome 开启远程桌面功能

sudo apt install gnome-remote-desktop ----长话短说 故障表现 Ubuntu 25版本点击远程桌面功能没有任何反应, WIN_20250630_00_53_24_Pro 最后 armbian 官方社区充满了傲慢,一言不合就关闭话题,问题都没有解决就给我关闭了 最后检索到英文网站,说到了这么一句话,检查远程桌…

嵌入式 Linux 入门:从裸机到系统级开发的第一步

随着嵌入式系统应用的不断深入&#xff0c;很多 MCU 项目开发者会在某个阶段遇到瓶颈&#xff1a;系统越来越复杂、任务越来越多、通信越来越频繁、性能要求越来越高。 这时候&#xff0c;从 MCU / RTOS 过渡到 嵌入式 Linux 开发 就成为一次技术升级的关键转折点。 本文将带…

详解 Blazor 组件传值

父子组件传值 在 Blazor 中&#xff0c;组件之间的通信可以通过 [Parameter] 参数和 EventCallback<T> 事件回调实现。下面分别给出 父组件传递值给子组件 和 子组件传递值给父组件 的简单示例。 1.1 父组件传递值给子组件 步骤&#xff1a; 在子组件中定义 public 属…

力扣热题100再刷

160.相交链表 读一遍A&#xff0c;一个set存节点&#xff0c;遍历B的时候判断即可。复习下set的STL&#xff1a;set有set和unordered_set&#xff0c;同样有insert&#xff0c;find&#xff0c;count&#xff0c;对于set而言&#xff0c;自动从小到大排序&#xff0c;还有&…

MySQL常用函数性能优化及索引影响分析

MySQL 常用函数性能优化指南&#xff08;含索引影响分析&#xff09; 以下是 MySQL 函数使用指南&#xff0c;新增性能影响评级、索引失效分析和优化方案&#xff0c;帮助您高效使用函数&#xff1a; &#x1f4dc; 一、字符串处理函数&#xff08;含性能分析&#xff09; 函…

莫队(基础版)优雅的暴力

莫队算法是一种离线算法&#xff0c;常用于高效处理区间查询问题。它通过合理排序和移动左右端点来减少时间复杂度。 基本思想 莫队算法的核心思想是将所有查询离线排序&#xff01;&#xff01;&#xff08;找出一个过起来最快的查询顺序&#xff09;&#xff0c;然后通过移动…

✨ Python 高级定制 | 美化 Word 表格边框与样式(收货记录增强版)

之前我们完成了 Excel 数据提取、Word 表格写入与合并&#xff0c;现在继续 为 Word 表格添加高级样式 装扮&#xff0c;包括单元格边框、背景填色、居中对齐、粗体、高亮行/列等&#xff0c;进一步增强表格的可读性与专业性。 &#x1f58c;️ 样式设置函数 1. 设置单元格边框…

Clickhouse源码分析-TTL执行流程

第一种情况&#xff1a;无ttl_only_drop_parts配置 总体示例以及说明 如果没有ttl_only_drop_parts的配置&#xff0c;过期数据的删除&#xff08;这里是删除&#xff0c;是将过期的数据从这个part删除&#xff0c;并将过期的数据构成一个part&#xff0c;这个过期的part标记…

elementui修改radio字体的颜色和圆圈的样式

改完 <div class"choose"><el-radio-group v-model"radioNum"><el-radio label"1" size"large">Option 1</el-radio><el-radio label"2" size"large">Option 2</el-radio>&l…

力扣3381. 长度可被 K 整除的子数组的最大元素和

由于数据范围是2*10^5所以必然是遍历一次&#xff0c;子数组必定要用到前缀和&#xff0c;之前的题目中总是遇到的是子数组的和能不能被k整除&#xff0c;而这里不一样的是子数组的长度能不能被k整除&#xff0c;如果单纯的枚举长度必定超时&#xff0c;而看看题解得出的思路&a…

基于SSM的勤工助学系统的设计与实现

第1章 摘要 基于SSM框架的勤工助学系统旨在为学生、用工部门和管理员提供高效便捷的管理平台。系统包括学生端、用工部门端和管理员端&#xff0c;涵盖了从岗位发布、申请审核、工时记录、薪资管理到数据统计等完整的功能需求。 学生可以通过系统首页浏览最新的岗位信息和公告&…

2025年06月30日Github流行趋势

项目名称&#xff1a;twenty 项目地址 URL&#xff1a;https://github.com/twentyhq/twenty项目语言&#xff1a;TypeScript历史 star 数&#xff1a;31,774今日 star 数&#xff1a;1,002项目维护者&#xff1a;charlesBochet, lucasbordeau, FelixMalfait, Weiko, bosiraphae…

creo 2.0学习笔记

Creo软件从入门到精通——杜书森 1.1 Creo基本建模过程介绍 新建-零件-改名称-取消使用默认模板&#xff0c;是因为默认的是英制尺寸&#xff0c;自定义可选择mmns_part_solid&#xff0c;模板主要是设置模型的单位拉伸-选取FRONT-点击草绘视图&#xff0c;可进行草绘旋转——…

ZNS初步认识—GPT

1. ZNS SSD 的基本概念 Zoned Namespace (ZNS): ZNS 是一种新的NVMe接口规范&#xff0c;它将SSD的逻辑块地址空间划分为多个独立的、固定大小的“区域”&#xff08;Zones&#xff09;。区域 (Zone): ZNS SSD 的基本管理单元。每个区域都有自己的写入指针&#xff08;write p…