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

RocketMQ顺序消费机制

RocketMQ的顺序消费机制通过生产端和消费端的协同设计实现,其核心在于局部顺序性,即保证同一队列(MessageQueue)内的消息严格按发送顺序消费。以下是详细机制解析及关键源码实现:
在这里插入图片描述


一、顺序消费的核心机制

1. 生产端路由策略
  • Sharding Key路由:生产者通过MessageQueueSelector接口将同一业务标识(如订单ID)的消息路由到同一队列。例如,根据订单ID对队列数取模,确保同一订单的消息进入同一队列。
    // 示例:生产者选择队列
    SendResult sendResult = producer.send(msg, new MessageQueueSelector() {
      @Override
      public MessageQueue select(List<MessageQueue> mqs, Message msg, Object arg) {
         Integer id = (Integer) arg;
         int index = id % mqs.size();
         return mqs.get(index);
      }
    }, orderId);
    

路由方法:
在这里插入图片描述
SelectMessageQueueByHash:按哈希选择消息队列。
SelectMessageQueueByRandom:随机选择消息队列。
SelectMessageQueueByMachineRoom:按照机房选择消息队列。

  • 同步发送:必须使用同步发送(send()方法),异步发送无法保证消息顺序。
2. 消费端锁机制
  • Broker端队列锁:消费者集群模式下,通过定时任务(默认每20秒)向Broker申请队列锁,只有获得锁的消费者实例才能拉取并消费该队列消息。锁的有效期默认60秒,避免宕机导致死锁。
  • 本地队列快照锁:消费者在消费时对ProcessQueue(队列快照)加内存锁(synchronized块),确保同一队列的消息仅由一个线程顺序处理。
3. 消费流程控制
  • 单线程顺序消费:每个队列对应一个消费线程,从ProcessQueue的红黑树(msgTreeMap)中按消息偏移量顺序取出消息,保证消费顺序与存储顺序一致。
  • 失败重试机制:消费失败时,若未达最大重试次数,消息会重新放回ProcessQueue等待下次消费;若超过次数则进入死信队列。

二、关键源码解析

1. 消费者启动与锁管理
  • 服务初始化:消费者启动时,若监听器为MessageListenerOrderly,则创建ConsumeMessageOrderlyService,并启动定时加锁任务。

    // DefaultMQPushConsumerImpl#start
    if (getMessageListenerInner() instanceof MessageListenerOrderly) {
        this.consumeMessageService = new ConsumeMessageOrderlyService(this, listener);
        consumeMessageService.start();
    }
    
  • 定时加锁ConsumeMessageOrderlyService启动后,定时调用RebalanceImpl.lockAll()向Broker申请锁,更新ProcessQueue的锁定状态。

      public synchronized void lockMQPeriodically() {
          if (!this.stopped) {
              this.defaultMQPushConsumerImpl.getRebalanceImpl().lockAll();
          }
      }
    
      				for (MessageQueue mq : mqs) {
                          ProcessQueue processQueue = this.processQueueTable.get(mq);
                          if (processQueue != null) {
                              if (lockOKMQSet.contains(mq)) {
                                  if (!processQueue.isLocked()) {
                                      log.info("the message queue locked OK, Group: {} {}", this.consumerGroup, mq);
                                  }
                                 // 更新`ProcessQueue`的锁定状态 true
                                  processQueue.setLocked(true);
                                  processQueue.setLastLockTimestamp(System.currentTimeMillis());
                              } else {
                                 // 更新`ProcessQueue`的锁定状态 false
                                  processQueue.setLocked(false);
                                  log.warn("the message queue locked Failed, Group: {} {}", this.consumerGroup, mq);
                              }
                          }
                      }
    
2. 消息拉取与消费
  • 锁检查:拉取消息前检查ProcessQueue是否已锁定,未锁定则延迟拉取。
    // DefaultMQPushConsumerImpl#pullMessage
    if (processQueue.isLocked()) {
        // 计算消费偏移量并拉取消息
    } else {
        executePullRequestLater(pullRequest, 3000); // 延迟3秒重试
    }
    
  • 消费线程加锁:消费线程运行时获取队列内存锁,确保单线程处理。
    synchronized (messageQueueLock.fetchLockObject(messageQueue)) {
        List<MessageExt> msgs = processQueue.takeMessags(batchSize);
        // 执行消费逻辑
    }
    
3. Broker端锁管理
  • 锁存储:Broker通过RebalanceLockManager维护锁信息,记录消费者ClientID和最后更新时间,超时(默认60秒)则自动释放。
    class LockEntry {
        String clientId;
        long lastUpdateTimestamp;
        boolean isExpired() { /* 检查是否超时 */ }
    }
    
  • 锁竞争:消费者通过lockBatchMQ请求批量加锁,Broker返回成功锁定的队列列表。

三、适用场景与注意事项

  1. 适用场景

    • 分区顺序:如订单流程(创建、支付、完成),同一订单ID的消息需顺序处理。
    • 全局顺序Topic仅一个队列,性能较低,适用于强一致性场景(如证券交易)。
  2. 注意事项

    • 幂等性:因网络抖动或消费者重启可能导致短暂乱序,业务逻辑需支持幂等处理。
    • 队列数选择:分区数越多并发度越高,但需确保同一业务ID的路由一致性。

总结

RocketMQ的顺序消费通过生产端路由策略消费端锁机制Broker协同管理实现。其设计在保证局部顺序的同时兼顾性能,适用于多数业务场景。源码层面,ConsumeMessageOrderlyServiceRebalanceImpl是核心模块,通过定时加锁单线程消费队列快照管理确保顺序性。实际使用时需结合业务特点设计Sharding Key,并处理可能的异常情况。

相关文章:

  • 智慧水务新时代:1.05亿项目引领的数字化浪潮
  • 从“0”开始入门PCB之(5)完结篇!--快速入门原理图DRC,PCB的符号与布局,2D与3D效果,PCB的图层和布线,PCB板框
  • 备赛蓝桥杯之第十五届职业院校组省赛第六题:简易JSX解析器
  • 深入探索WebGL:解锁网页3D图形的无限可能
  • AI控制游戏 -- 开源框架来了!
  • 电力MOSFET的输出特性与转移特性
  • 【基于手势识别的音量控制系统】
  • Kali GRUB 修复
  • 19c rac-expdp备份异常处理
  • 使用Perl和库WWW::Curl的爬虫程序
  • 解锁数据潜能,永洪科技以数据之力简化中粮可口可乐决策之路
  • 001-Boost消息队列的使用
  • python学习笔记-day5
  • CuPy加速计算 —— 使用 Python 进行 GPU 加速计算的 NumPy/SciPy 兼容数组库
  • 贝叶斯定理全解析
  • Springboot 梳理
  • 理解 Python 中的 `asyncio` 模块
  • python绘图之双轴图
  • leetcode5:最长回文子串
  • 利用golang embed特性嵌入前端资源问题解决
  • 国内哪家网站做的系统纯净/百度官方网站下载
  • 莆田手表网站/怎么安装百度
  • 做网站温州/排名优化seo
  • wordpress建站很麻烦/企业网站关键词优化
  • 网站开发公司售后服务/seo网站排名厂商定制
  • phpcms v9 网站建设设计制作网络科技模板/免费b站推广网站在线