RocketMQ 延迟消息解析——图解、源码级解析

本文涉及的产品
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: RocketMQ 延迟消息解析——图解、源码级解析


定义

**<font color="#33aaff">延迟消息</font>:** 生产者发送消息之后,不能立刻被消费者消费,而是应该等到一定时间之后才可以被消费。


**例如:** 在淘宝购物时,当用户提交一个订单之后,如果30min还没有支付的话,就给用户发送一条提示消息


Broker提供延迟消息功能的核心思路是一致的:将延迟消息先存放在一个临时的空间里,等到期后才发送给Topic


![在这里插入图片描述](https://ucc.alicdn.com/images/user-upload-01/a8f10b51615a462795e6363f4da2cea6.png#pic_Center)

如上图所示,实现延迟消息的步骤如下:

1. Producer发送一个延迟消息

2. Broker将其通过临时存储进行存储

3. 延迟服务(delay service)检查消息持否到期,将到期的消息投递到目标Topic里

4. 消费者消费延迟消息


根据上面的流程不难发现,临时存储和延迟服务是实现延迟消息的关键,并且外部的Producer和Consumer是感知不到的


> RocketMQ将延迟消息存储在一个内部主题`SCHEDULE_TOPIC_XXXX`里,只支持固定的时间精度,如5s、10s、1min等


> Kafka原生不支持延迟消息,此时就可以借助代理 + 第三方存储来实现延迟消息


如果要借助第三方存储服务,则在技术选型时需要满足一下几个条件:

**1. 高性能:** 写入延迟要低,MQ的一个重要作用是削峰填谷,所以在高峰期写入性能一定要高,MySQL就不满足


**2. 高可靠:** 延迟消息写入之后,不能丢失


**3. 支持排序:** 要支持按照时间先后来排序,例如先发一条10s的延迟消息A,再发一条5s的延迟消息B,B要比A先消费掉


<br/>


# DDMQ

DDMQ就使用了代理 + 第三方存储来实现延迟消息:

![在这里插入图片描述](https://ucc.alicdn.com/images/user-upload-01/9ddf66d631bd449e9e6b6c399335ee3b.png?x-oss-process=image/watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBA5bCP546L5pu-5piv5bCR5bm0,size_18,color_FFFFFF,t_70,g_se,x_16#pic_center)

上图的第三方存储技术选型是`rocksdb`,它是一个高性能KV存储(类似于Redis),并且支持排序


**执行步骤:**

1. 生产者将延迟消息发送给代理,代理将其发送到一个缓冲Topic中

2. 消息存储在`rocksdb`里,以时间为key

3. delay service判断消息时间到期后就将其发送到目标Topic里

4. 消费者消费Topic里的数据


<br/>


不难发现,这种设计体现了解耦的思路:


- delay service的延迟投递能力是独立于Broker实现的,无序对Broker做任何改造,所以理论上对于任意MQ类型都可以提供延迟消息能力


## delay service的设计细节

- **高可用:** delay service以分布式部署了多个节点

- **最终一致性:** 每个delay service都要消费缓冲Topic的全量数据,这样就有了多个备份,可以保证数据不丢失

- **避免重复消费:** delay service选择一个主节点Master将消息投递到目标Topic里,这一步可以借助zookeeper来实现

- Master需要将自己当前已经投递到的时间记录到一个共享存储里,如果Master挂了,则会重新选举出一个新的Master,从之前的时间点继续开始投递


<br/><br/>

# RocketMQ中的延迟消息

RocketMQ默认只支持18个时间延迟消息,由配置项`messageDelayLevel`决定:


```java

messageDelayLevel=1s 5s 10s 30s 1m 2m 3m 4m 5m 6m 7m 8m 9m 10m 20m 30m 1h 2h

```


Broker每个节点在启动时都会创建18个级别的队列,如果想要修改,直接在安装了RocketMQ的机器上修改配置文件即可。


## 生产者发送延迟消息

只需要设置一下延迟级别,延迟级别对应上面`messageDelayLevel`的索引:


```java

Message msg = new Message();

msg.setTopic("Topic A");

msg.setTags("Tag A");

msg.setBody("delay message".getBytes());

// 设置延迟级别 = 1,对应延迟1s

msg.setDelayTimeLevel(1);

```

如果设置的延迟级别超过最大值,则会自动设置成最大值


## Broker端处理延迟消息

Broker端处理延迟消息一共分为5个步骤,如下图所示:

![在这里插入图片描述](https://ucc.alicdn.com/images/user-upload-01/78d207e36e34438ea47c474e10bd8e81.png?x-oss-process=image/watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBA5bCP546L5pu-5piv5bCR5bm0,size_16,color_FFFFFF,t_70,g_se,x_16#pic_center)


**整个过程分为5个步骤:**

1. 生产者发送延迟消息到Broker里

2. 把消息转发到`SCHEDULE_TOPIC_XXXX`主题下的队列中

3. 延迟服务定期消费`SCHEDULE_TOPIC_XXXX`主题下的消息,到时间了就把它拿到`CommitLog`中

4. 消息重新被投放到目标`Topic`里

5. 消费者消费延迟消息


<hr/>


**1. 生产者发送延迟消息到Broker里**


```java

public PutMessageResult putMessage(final MessageExtBrokerInner msg) {


   //事务消息处理

   if (tranType == MessageSysFlag.TRANSACTION_NOT_TYPE

           || tranType == MessageSysFlag.TRANSACTION_COMMIT_TYPE) {

       // 如果是延迟消息

       if (msg.getDelayTimeLevel() > 0) {

           // 如果设置的值过大,则设置为最大延迟级别

           if (msg.getDelayTimeLevel() > this.defaultMessageStore.getScheduleMessageService().getMaxDelayLevel()) {

               msg.setDelayTimeLevel(this.defaultMessageStore.getScheduleMessageService().getMaxDelayLevel());

           }


           // 修改Topic

           topic = ScheduleMessageService.SCHEDULE_TOPIC;

           // 根据延迟级别,决定要将其投递到那个队列中

           queueId = ScheduleMessageService.delayLevel2QueueId(msg.getDelayTimeLevel());


           // 记录原始的 topic 和 队列信息

           MessageAccessor.putProperty(msg, MessageConst.PROPERTY_REAL_TOPIC, msg.getTopic());

           MessageAccessor.putProperty(msg, MessageConst.PROPERTY_REAL_QUEUE_ID, String.valueOf(msg.getQueueId()));

           msg.setPropertiesString(MessageDecoder.messageProperties2String(msg.getProperties()));


           // 修改topic和队列信息

           msg.setTopic(topic);

           msg.setQueueId(queueId);

       }

   }


   // ......


}

```


可以看到,RocketMQ的Broker端在存储生产者写入消息时,首先将其写入`CommitLog`里,为了不让用户立刻就能消费到这条消息,这里先将`Topic`的名称修改为`SCHEDULE_TOPIC_XXXX`,并且根据设置的延迟级别选择将消息投放到哪一个队列里。


在修改`Topic`的名称时,还会将原来要发送到的`Topic`和队列信息存储在消息的属性里


<br/>


**2. 把消息转发到`SCHEDULE_TOPIC_XXXX`主题下的队列中**


这一步的操作是异步执行的,在转发过程中会对延迟消息进行特殊处理,主要是计算这条消息什么时间要被投递到最终的`Topic`里


> 投递时间  = 存储消息的世界 + 延迟时间


ConsumeQueue的单个存储结构如图所示:

![在这里插入图片描述](https://ucc.alicdn.com/images/user-upload-01/a89e9fdca9dc4427a4f7482f7d5c21ea.png#pic_center)

- `Commit Log Offset`:消息在Commit中的存放位置

- `Size`:消息体大小

- `Message Tag HashCode`:消息Tag的hash值记录在此。对于延迟消息,这里记录的是消息的投递时间,<font color = "#ff3333">所以才是8Byte而不是4Byte</font>


```java

public DispatchRequest checkMessageAndReturnSize(java.nio.ByteBuffer byteBuffer, final boolean checkCRC, final boolean readBody) {

    // ......

       try {

        // ......

               { // 获取延迟级别

                   String t = propertiesMap.get(MessageConst.PROPERTY_DELAY_TIME_LEVEL);

                   // 如果消息需要投递到`SCHEDULE_TOPIC_XXXX`主题里

                   if (ScheduleMessageService.SCHEDULE_TOPIC.equals(topic) && t != null) {

                       int delayLevel = Integer.parseInt(t);

                       // 延迟级别大于0,则计算目标投递时间,并将其记录在Message Tag HashCode里

                       if (delayLevel > 0) {

                           tagsCode = this.defaultMessageStore.getScheduleMessageService().computeDeliverTimestamp(delayLevel,

                                   storeTimestamp);

                       }

                   }

               }

           }

           // ....

           return new DispatchRequest(//

               topic, // 1

               queueId, // 2

               physicOffset, // 3

               totalSize, // 4

               tagsCode, // 5

               storeTimestamp, // 6

               queueOffset, // 7

               keys, // 8

               uniqKey, //9

               sysFlag, // 9

               preparedTransactionOffset// 10

           );

       } catch (Exception e) {

       }


       return new DispatchRequest(-1, false /* success */);

   }

```


<br/>


**3. 延迟服务定期消费`SCHEDULE_TOPIC_XXXX`主题下的消息,到时间了就把它拿到`CommitLog`中**


Broker内部的`ScheduleMessageService`类提供延迟服务,该服务消费`SCHEDULE_TOPIC_XXXX`主题下的消息,并投递到目标`Topic`下。


```java

public void start() {

       if (started.compareAndSet(false, true)) {

        // 创建定时器

           this.timer = new Timer("ScheduleMessageTimerThread", true);

           // 循环所有的延迟级别,delayLevelTable记录每个延迟级别对应的延迟时间

           for (Map.Entry<Integer, Long> entry : this.delayLevelTable.entrySet()) {

               Integer level = entry.getKey();

               Long timeDelay = entry.getValue();

               Long offset = this.offsetTable.get(level);

               if (null == offset) {

                   offset = 0L;

               }

   // 针对每个延迟级别创建一个TimerTask

               if (timeDelay != null) {

                   this.timer.schedule(new DeliverDelayedMessageTimerTask(level, offset), FIRST_DELAY_TIME);

               }

           }


           this.timer.scheduleAtFixedRate(new TimerTask() {


               @Override

               public void run() {

                   try {

                       if (started.get()) ScheduleMessageService.this.persist();

                   } catch (Throwable e) {

                       log.error("scheduleAtFixedRate flush exception", e);

                   }

               }

           }, 10000, this.defaultMessageStore.getMessageStoreConfig().getFlushDelayOffsetInterval());

       }

   }

```

可以看出,`ScheduleMessageService`启动时会启动一个`Timer`,并根据延迟级别的个数创建对应数量的`TimerTask`,每个`TimerTask`负责一个延迟级别中的消息的投递。


<font color="#aaaaff">每个`TimerTask`在检查消息是否到期时,首先会检查对应队列中尚未投递的第一条消息,如果这条消息没到期,说明后面的消息都没到期,就不用再做进一步的检查了。</font>


<br/>


在第1步执行时,已经存下了原先要投递到的目标`TOPIC`和队列信息,这里只需要重新设置一下即可。此外,在第2步的执行过程里`Message Tag HashCode`存储的是消息的投递时间,这里要重新计算成生产者设置的Tag的hash值。


<br/>


**4. 消息重新被投放到目标`Topic`里**

与第2步类似,只不过这里的`Topic`已经改成了目标`Topic`,此时消费者就可以消费到这条消息


<br/><br/>

# 延迟消息与消息重试机制的关系

RocketMQ提供了消息消费失败重试的机制,当消息被消费失败时,如果返回的值是`RECONSUME_LATER`的话就会重试:


```java

DefaultMQPushConsumer consumer = new DefaultMQPushConsumer();


consumer.registerMessageListener(new MessageListenerConcurrently() {

   public ConsumeConcurrentlyStatus consumeMessage(List<MessageExt> list, ConsumeConcurrentlyContext consumeConcurrentlyContext) {

       return ConsumeConcurrentlyStatus.RECONSUME_LATER;

   }

});

```

默认的重试次数是16次,并且会随着失败次数的增加,重试的时间间隔也会增加,具体如下表:

| 重试次数 | 时间间隔 |重试次数 |时间间隔 |重试次数 |时间间隔 |重试次数 |时间间隔 |

|--|--|--|--|--|--|--|--|

| 1 | 10s |5 | 3min|9 | 7min| 13|20min |

|2  |30s  |6 |4min | 10|8min | 14|30min |

| 3 | 1min |7 |5min |11 |9min | 15|1h |

| 4 | 2min | 8|6min |12 |10min |16 | 2h|


实际上,消息重试的时间间隔,就对应延迟消息的后16个级别:



messageDelayLevel=1s5s10s30s1m2m3m4m5m6m7m8m9m10m20m30m1h2h

原因是消息重试的底层也是借助延迟消息来实现的,在消费失败的情况下会重新把消息当做延迟消息投递回去,官方文档也给了说明:

![在这里插入图片描述](https://ucc.alicdn.com/images/user-upload-01/29489bfab75c4e0bb2c84b615aa798f7.png?x-oss-process=image/watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBA5bCP546L5pu-5piv5bCR5bm0,size_20,color_FFFFFF,t_70,g_se,x_16#pic_center)

相关实践学习
消息队列RocketMQ版:基础消息收发功能体验
本实验场景介绍消息队列RocketMQ版的基础消息收发功能,涵盖实例创建、Topic、Group资源创建以及消息收发体验等基础功能模块。
消息队列 MNS 入门课程
1、消息队列MNS简介 本节课介绍消息队列的MNS的基础概念 2、消息队列MNS特性 本节课介绍消息队列的MNS的主要特性 3、MNS的最佳实践及场景应用 本节课介绍消息队列的MNS的最佳实践及场景应用案例 4、手把手系列:消息队列MNS实操讲 本节课介绍消息队列的MNS的实际操作演示 5、动手实验:基于MNS,0基础轻松构建 Web Client 本节课带您一起基于MNS,0基础轻松构建 Web Client
相关文章
|
2月前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
94 2
|
3月前
|
缓存 Java 程序员
Map - LinkedHashSet&Map源码解析
Map - LinkedHashSet&Map源码解析
89 0
|
3月前
|
算法 Java 容器
Map - HashSet & HashMap 源码解析
Map - HashSet & HashMap 源码解析
69 0
|
16天前
|
存储 设计模式 算法
【23种设计模式·全精解析 | 行为型模式篇】11种行为型模式的结构概述、案例实现、优缺点、扩展对比、使用场景、源码解析
行为型模式用于描述程序在运行时复杂的流程控制,即描述多个类或对象之间怎样相互协作共同完成单个对象都无法单独完成的任务,它涉及算法与对象间职责的分配。行为型模式分为类行为模式和对象行为模式,前者采用继承机制来在类间分派行为,后者采用组合或聚合在对象间分配行为。由于组合关系或聚合关系比继承关系耦合度低,满足“合成复用原则”,所以对象行为模式比类行为模式具有更大的灵活性。 行为型模式分为: • 模板方法模式 • 策略模式 • 命令模式 • 职责链模式 • 状态模式 • 观察者模式 • 中介者模式 • 迭代器模式 • 访问者模式 • 备忘录模式 • 解释器模式
【23种设计模式·全精解析 | 行为型模式篇】11种行为型模式的结构概述、案例实现、优缺点、扩展对比、使用场景、源码解析
|
16天前
|
设计模式 存储 安全
【23种设计模式·全精解析 | 创建型模式篇】5种创建型模式的结构概述、实现、优缺点、扩展、使用场景、源码解析
结构型模式描述如何将类或对象按某种布局组成更大的结构。它分为类结构型模式和对象结构型模式,前者采用继承机制来组织接口和类,后者釆用组合或聚合来组合对象。由于组合关系或聚合关系比继承关系耦合度低,满足“合成复用原则”,所以对象结构型模式比类结构型模式具有更大的灵活性。 结构型模式分为以下 7 种: • 代理模式 • 适配器模式 • 装饰者模式 • 桥接模式 • 外观模式 • 组合模式 • 享元模式
【23种设计模式·全精解析 | 创建型模式篇】5种创建型模式的结构概述、实现、优缺点、扩展、使用场景、源码解析
|
16天前
|
设计模式 存储 安全
【23种设计模式·全精解析 | 创建型模式篇】5种创建型模式的结构概述、实现、优缺点、扩展、使用场景、源码解析
创建型模式的主要关注点是“怎样创建对象?”,它的主要特点是"将对象的创建与使用分离”。这样可以降低系统的耦合度,使用者不需要关注对象的创建细节。创建型模式分为5种:单例模式、工厂方法模式抽象工厂式、原型模式、建造者模式。
【23种设计模式·全精解析 | 创建型模式篇】5种创建型模式的结构概述、实现、优缺点、扩展、使用场景、源码解析
|
2月前
|
缓存 监控 Java
Java线程池提交任务流程底层源码与源码解析
【11月更文挑战第30天】嘿,各位技术爱好者们,今天咱们来聊聊Java线程池提交任务的底层源码与源码解析。作为一个资深的Java开发者,我相信你一定对线程池并不陌生。线程池作为并发编程中的一大利器,其重要性不言而喻。今天,我将以对话的方式,带你一步步深入线程池的奥秘,从概述到功能点,再到背景和业务点,最后到底层原理和示例,让你对线程池有一个全新的认识。
60 12
|
1月前
|
PyTorch Shell API
Ascend Extension for PyTorch的源码解析
本文介绍了Ascend对PyTorch代码的适配过程,包括源码下载、编译步骤及常见问题,详细解析了torch-npu编译后的文件结构和三种实现昇腾NPU算子调用的方式:通过torch的register方式、定义算子方式和API重定向映射方式。这对于开发者理解和使用Ascend平台上的PyTorch具有重要指导意义。
|
2月前
|
消息中间件 存储 监控
深度写作:深入源码理解MQ长轮询优化机制
【11月更文挑战第22天】在分布式系统中,消息队列(Message Queue, MQ)扮演着至关重要的角色。MQ不仅实现了应用间的解耦,还提供了异步消息处理、流量削峰等功能。而在MQ的众多特性中,长轮询(Long Polling)机制因其能有效提升消息处理的实时性和效率,备受关注。
78 12
|
17天前
|
安全 搜索推荐 数据挖掘
陪玩系统源码开发流程解析,成品陪玩系统源码的优点
我们自主开发的多客陪玩系统源码,整合了市面上主流陪玩APP功能,支持二次开发。该系统适用于线上游戏陪玩、语音视频聊天、心理咨询等场景,提供用户注册管理、陪玩者资料库、预约匹配、实时通讯、支付结算、安全隐私保护、客户服务及数据分析等功能,打造综合性社交平台。随着互联网技术发展,陪玩系统正成为游戏爱好者的新宠,改变游戏体验并带来新的商业模式。

推荐镜像

更多
下一篇
开通oss服务