0

0

Java面试——消息队列中如何保证消息不丢失

畫卷琴夢

畫卷琴夢

发布时间:2026-01-30 08:23:02

|

468人浏览过

|

来源于php中文网

原创

RabbitMQ消息可靠性需四层保障:①启用生产者确认机制(confirmSelect+waitForConfirms或ConfirmListener);②交换机、队列、消息三者均设为持久化;③消费者关闭autoAck,手动调用basicAck;④应用层引入消息状态表与定时补偿机制。

java面试——消息队列中如何保证消息不丢失

开启生产者确认机制(publisher-confirm)

RabbitMQ 默认不保证消息一定到达交换机,网络抖动、Broker 重启、连接中断都可能导致 basicPublish 调用成功但消息实际未送达。必须显式启用发布确认:

channel.confirmSelect(); // 启用确认模式
channel.basicPublish("exchange", "routingKey", MessageProperties.PERSISTENT_TEXT_PLAIN, "data".getBytes());
if (!channel.waitForConfirms(5000)) { // 等待 ACK,超时抛异常
    throw new RuntimeException("消息未被交换机接收");
}

常见错误是只调用 basicPublish 就认为“发出去了”,没等 waitForConfirms 或忽略返回值。更稳妥的做法是配合异步回调(addConfirmListener),尤其在高并发场景下避免阻塞线程。

强制消息与队列持久化

即使消息到了交换机,若未正确路由或未落盘,仍可能丢失。三个环节必须同时持久化:

  • 交换机声明时设 durable=true
  • 队列声明时设 durable=true
  • 发送消息时使用 MessageProperties.PERSISTENT_TEXT_PLAIN(即 deliveryMode=2)

缺一不可。例如只设队列持久化但消息是非持久的,Broker 内存压力大时会丢弃该消息;又或者交换机非持久,Broker 重启后交换机消失,后续消息直接被丢弃且无报错。

立即学习Java免费学习笔记(深入)”;

消费者手动 ACK + 关闭 autoAck

消费者端最容易踩的坑是依赖默认的 autoAck=true。一旦消息投递后消费者进程崩溃(如 OOM、kill -9),消息就永久丢失——Broker 认为它已被“自动签收”。

必须配置:

MiroThinker
MiroThinker

MiroMind团队推出的研究型开源智能体,专为深度研究与复杂工具使用场景设计

下载
channel.basicConsume("queue", false, consumer); // 第二个参数为 false

并在业务逻辑完全执行成功后,再调用:

channel.basicAck(deliveryTag, false);

注意:如果处理逻辑涉及数据库写入,要确保 DB 提交成功后再发 basicAck;否则 DB 回滚但消息已标记为消费,造成数据不一致。

兜底方案:消息状态表 + 定时补偿

RabbitMQ 自身机制能覆盖绝大多数异常,但无法应对极端情况:比如生产者写完 DB 后、发消息前宕机;或消费者处理成功、发 ACK 前网络断开。

这时需要应用层兜底:

  • 发送方先插入一条记录到 msg_status 表,字段含 msg_idcontentstatus='pending'send_time
  • 再调用 basicPublish
  • 消费方处理完,通过 HTTP/Feign 回调发送方接口,将对应 msg_id 的状态更新为 'consumed'
  • 后台定时任务扫描 status='pending'send_time 超过 5 分钟的消息,重新投递

这个表不是可选优化,而是金融、订单类场景的硬性要求。很多人以为“开了持久化+确认就够了”,却在灰度发布时因一次磁盘故障发现几百条订单消息石沉大海——那正是没做状态对账的代价。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
rabbitmq和kafka有什么区别
rabbitmq和kafka有什么区别

rabbitmq和kafka的区别:1、语言与平台;2、消息传递模型;3、可靠性;4、性能与吞吐量;5、集群与负载均衡;6、消费模型;7、用途与场景;8、社区与生态系统;9、监控与管理;10、其他特性。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

202

2024.02.23

Java 消息队列与异步架构实战
Java 消息队列与异步架构实战

本专题系统讲解 Java 在消息队列与异步系统架构中的核心应用,涵盖消息队列基本原理、Kafka 与 RabbitMQ 的使用场景对比、生产者与消费者模型、消息可靠性与顺序性保障、重复消费与幂等处理,以及在高并发系统中的异步解耦设计。通过实战案例,帮助学习者掌握 使用 Java 构建高吞吐、高可靠异步消息系统的完整思路。

11

2026.01.28

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1135

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

213

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

1893

2025.12.29

java接口相关教程
java接口相关教程

本专题整合了java接口相关内容,阅读专题下面的文章了解更多详细内容。

21

2026.01.19

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

523

2023.08.10

数据库三范式
数据库三范式

数据库三范式是一种设计规范,用于规范化关系型数据库中的数据结构,它通过消除冗余数据、提高数据库性能和数据一致性,提供了一种有效的数据库设计方法。本专题提供数据库三范式相关的文章、下载和课程。

359

2023.06.29

C++ 设计模式与软件架构
C++ 设计模式与软件架构

本专题深入讲解 C++ 中的常见设计模式与架构优化,包括单例模式、工厂模式、观察者模式、策略模式、命令模式等,结合实际案例展示如何在 C++ 项目中应用这些模式提升代码可维护性与扩展性。通过案例分析,帮助开发者掌握 如何运用设计模式构建高质量的软件架构,提升系统的灵活性与可扩展性。

8

2026.01.30

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 3万人学习

C# 教程
C# 教程

共94课时 | 8万人学习

Java 教程
Java 教程

共578课时 | 53.5万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号