RabbitMQ 消息顺序、消息幂等、消息重复、消息事务、集群

本文探讨了消息系统中的消息顺序、幂等性、重复处理等问题,并对比了RabbitMQ和Kafka的特点,提供了应用层解决策略。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1. 消息顺序

场景:比如下单操作,下单成功之后,会发布创建订单和扣减库存消息,但扣减库存消息执行会先于创建订单消息,也就说前者执行成功之后,才能执行后者。

不保证完全按照顺序消费,在 MQ 层面支持消息的顺序处理开销太大,为了极少量的需求,增加整体上的复杂度得不偿失。

所以,还是在应用层面处理比较好,或者业务逻辑进行处理

应用层解决方式:

  • 1. 消息实体中增加:版本号 & 状态机 & msgid & parent_msgid,通过 parent_msgid 判断消息的顺序(需要全局存储,记录消息的执行状态)。
  • 2. “同步执行”:当一个消息执行完之后,再发布下一个消息。

2. 消息幂等、消息重复、消息事务

消息重复

造成消息重复的根本原因是:网络不可达。只要通过网络交换数据,就无法避免这个问题。所以解决这个问题的办法就是绕过这个问题。那么问题就变成了:如果消费端收到两条一样的消息,应该怎样处理?

消费端处理消息的业务逻辑保持幂等性。

保证每条消息都有唯一编号且保证消息处理成功与去重表的日志同时出现。

第 1 条很好理解,只要保持幂等性,不管来多少条重复消息,最后处理的结果都一样。第 2 条原理就是利用一张日志表来记录已经处理成功的消息的 ID,如果新到的消息 ID 已经在日志表中,那么就不再处理这条消息。

第 1 条解决方案,很明显应该在消费端实现,不属于消息系统要实现的功能。第 2 条可以消息系统实现,也可以业务端实现。正常情况下出现重复消息的概率其实很小,如果由消息系统来实现的话,肯定会对消息系统的吞吐量和高可用有影响,所以最好还是由业务端自己处理消息重复的问题,这也是 RabbitMQ 不解决消息重复的问题的原因。

RabbitMQ 不保证消息不重复,如果你的业务需要保证严格的不重复消息,需要你自己在业务端去重。

AMQP 消费者确认机制

AMQP 定义了消费者确认机制(message ack),如果一个消费者应用崩溃掉(此时连接会断掉,broker 会得知),但是 broker 尚未获得 ack,那么消息会被重新放入队列。所以 AMQP 提供的是“至少一次交付”(at-least-once delivery),异常情况下,消息会被重复消费,此时业务要实现幂等性(重复消息处理)。

AMQP 生产者事务

对于生产者,AMQP 定义了事务(tx transaction)来确保生产消息被 broker 接收并成功入队。TX 事务是阻塞调用,生产者需等待broker写磁盘后返回的确认,之后才能继续发送消息。事务提交失败时(如broker宕机场景),broker并不保证提交的消息全部入队。

TX 的阻塞调用使 broker 的性能非常差,RabbitMQ 使用 confirm 机制来优化生产消息的确认。Confirm 模式下,生产者可以持续发送消息,broker 将消息批量写磁盘后回复确认,生产者通过确认消息的ID来确定哪些已发送消息被成功接收。Confirm 模式下生产者发送消息和接受确认是异步流程,生产者需要缓存未确认的消息以便出错时重新发送。

总结

  • 1. 消息重复发布:不存在,因为 AMQP 定义了事务(tx transaction)来确保生产消息被 broker 接收并成功入队。TX 事务是阻塞调用,生产者需等待 broker 写磁盘后返回的确认,之后才能继续发送消息。事务提交失败时(如 broker 宕机场景),broker 并不保证提交的消息全部入队。RabbitMQ 使用 confirm 机制来优化生产消息的确认(可以持续发布消息,但会批量回复确认)。
  • 2. 消息重复消费:AMQP 提供的是“至少一次交付”(at-least-once delivery),异常情况下,消息会被重复消费,此时业务要实现幂等性(重复消息处理)。

应用层解决方式:

  • 1. 专门的 Map 存储:用来存储每个消息的执行状态(用 msgid 区分),执行成功之后更新 Map,有另外消息重复消费的时候,读取 Map 数据判断 msgid 对应的执行状态,已消费则不执行。
  • 2. 业务逻辑判断:消息执行完会更改某个实体状态,判断实体状态是否更新,如果更新,则不进行重复消费。

特别说明:AMQP 协议中的事务仅仅是指生产者发送消息给 broker 这一系列流程处理的事务机制,并不包含消费端的处理流程。

3. 集群

原 RabbitMQ 集群:manager1、manager2、manager3 节点均为磁盘存储,manager1 为主节点,HAProxy 负载三个节点。

现 RabbitMQ 集群更新(更合理的配置):

  • 1. RabbitMQ 集群更新:manager1、manager2 节点类型改为 ram(内存存储),manager3 节点类型为 disc(磁盘存储,用于保存集群配置和元数据),主节点变更为 manager3。
  • 2. HAProxy 负载更新:移除 manager3 负载(5672 端口),只保留 manage2、manager2 负载。

4. Kafka 和 RabbitMQ 对比

Kafka 的设计有明确的介绍:http://kafka.apache.org/documentation.html#design

Kafka 应对场景:消息持久化、吞吐量是第一要求、状态由客户端维护、必须是分布式的。Kafka 认为 broker 不应该阻塞生产者,高效的磁盘顺序读写能够和网络 IO 一样快,同时依赖现代 OS 文件系统特性,写入持久化文件时并不调用 flush,仅写入 OS pagecache,后续由 OS flush。

这些特性决定了 Kafka 没有做“确认机制”,而是直接将生产消息顺序写入文件、消息消费后不删除(避免文件更新),该实现充分利用了磁盘 IO,能够达到较高的吞吐量。代价是消费者要依赖 Zookeeper 记录队列消费位置、处理同步问题。没有消费确认机制,还导致了 Kafka 无法了解消费者速度,不能采用 push 模型以合理的速度向消费者推送数据,只能利用 pull 模型由消费者来拉消息(消费者承担额外的轮询开销)。

如果在 Kafka 中引入消费者确认机制,就需要 broker 维护消息消费状态,要做到高可靠就需要写文件持久化并与生产消息同步,这将急剧降低 Kafka 的性能,这种设计也极类似 RabbitMQ。如果不改变 Kafka 的实现,而是在 Kafka 和消费者之间做一层封装,还是需要实现一套类似 RabbitMQ 的消费确认和持久化机制。

参考资料:

### RabbitMQ 集群中保证消息顺序性的方法 在分布式系统中,RabbitMQ 是一种广泛使用的高性能消息中间件。然而,在某些特定场景下,需要确保消息按照发送的顺序被消费者接收并处理。以下是关于如何在 RabbitMQ 集群环境中实现消息顺序性的详细说明。 #### 1. 单生产者单队列模式 为了保证消息顺序性,最简单的做法是采用 **单生产者单队列** 的架构设计[^4]。在这种模式下,所有的消息都通过同一个 Producer 发送到同一个 Queue 中。由于 RabbitMQ 的内部机制决定了同一队列中的消息会被按序存储,因此只要消费者的数量不超过一个或者多个消费者之间能够协调好消费行为,则可以有效保障消息顺序性。 #### 2. 使用单一 Consumer 或串行化消费 当存在多 Consumer 场景时,即使所有消息都在同一个 Queue 中,也无法完全避免乱序现象的发生。这是因为 RabbitMQ 默认支持并发消费模型——即允许多个 Consumers 同时从同一个 Queue 获取数据。为了避免这种情况下的无序问题,可以选择以下两种策略之一: - 只部署一个 Consumer 实例来负责该 Queue 上的所有任务; - 如果确实需要用到多个实例提高性能,则需引入额外同步手段(比如基于 Redis 锁或其他分布式锁服务),使得每次只有一个实际执行操作的过程处于活动状态。 #### 3. 手动 Acknowledgment 结合事务管理 启用手动应答功能后,应用程序只有在其成功完成某条记录对应的业务逻辑之后才会向服务器返回确认信号 (ACK),从而减少因中途失败而导致的数据丢失风险。与此同时还可以考虑加入本地数据库写入等持久化措施作为补充保护层,进一步增强可靠性[^1]。 ```java // Java 示例代码展示如何设置手动 ACK 并进行安全的消息处理流程 Channel channel = connection.createChannel(); channel.basicQos(1); // 设置预取计数器为1,防止未处理完毕前再次分发新任务给当前客户端 DeliverCallback deliverCallback = (consumerTag, delivery) -> { String message = new String(delivery.getBody(), StandardCharsets.UTF_8); try{ processMessage(message); // 自定义函数模拟耗时较长的操作 System.out.println("Processed Message: " + message ); // 成功完成后才发出确认指令 channel.basicAck(delivery.getEnvelope().getDeliveryTag(), false); }catch(Exception e){ // 出错重试或丢弃依据具体需求决定 System.err.println(e.toString()); channel.basicNack(delivery.getEnvelope().getDeliveryTag(),false,true); } }; String queueName="example_queue"; boolean autoAck=false; channel.basicConsume(queueName ,autoAck ,deliverCallback ,(c)->{}); ``` #### 4. 考虑其他高级特性如优先级队列或延迟插件 虽然这些扩展并不直接针对解决全局意义上的有序传递诉求,但在特殊条件下它们也可能间接发挥作用帮助达成目标效果。例如利用 Priority Queues 让重要程度更高的项目总是先得到关注;又或者是借助 Delayed Messages Plugin 来安排未来某个时刻触发指定事件等等[^5]。 --- ### 总结 综上所述,在 RabbitMQ 集群环境下要实现严格意义下的消息顺序保持并非易事,往往涉及到架构层面的重大调整以及复杂度提升。推荐根据实际情况权衡利弊选取最适合自己的方案组合应用以上提到的各种技术要点。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值