TimerMessageStore 简略介绍

  • 延迟队列 rmq_sys_wheel_timer
  • 指定时间的延迟消息。会先投递到 rmq_sys_wheel_timer 队列中
  • 然后由 TimerMessageStore 消费队列数据,将数据消费到 timerWheel 使用时间轮算法,实现秒级任务

TimerMessageStore 操作的文件

  • store\consumequeue\rmq_sys_wheel_timer 从队列中读取消息, 提取数据存到 timerlogtimerwheel
  • store\checkpoint 对应 TimerMessageStore#timerCheckpoint
    • lastReadTimeMs 上次消费的时间节点
    • lastTimerLogFlushPos 最后刷新 log的 pos
    • lastTimerQueueOffset 最后一次消费的队列节点
    • masterTimerQueueOffset 主 Broker 的队列消费节点
  • store\timerwheel 时间轮,内由 Slot 组成 结构如下
    • timeMs 消息到达时间
    • firstPos 开始的 pos
    • lastPos 结束的 pos 在 timerLog 中读取数据, 后面会讲具体逻辑
    • num 消息数量
    • magic no use now, just keep it
  • store\timerlog 对应 TimerMessageStore#timerCheckpoint

    里边也是由多个 mappedFile 组成。

    主要是存储原msg的数据,

    因为从 rmq_sys_wheel_timer 消费了之后,

    会存到 timerwheeltimerlog

TimerMessageStore 启动

  • enqueueGetService.start();
  • enqueuePutService.start();
  • dequeueWarmService.start();
  • dequeueGetService.start();
  • timerFlushService.start();
  • dequeueGetMessageServices[getThreadNum].start();
  • dequeuePutMessageServices[getThreadNum].start();

深入 TimerMessageStore 之 TimerEnqueueGetService

  • TimerMessageStore.this.enqueue 默认 100毫秒执行一次
  • 从 消息队列 rmq_sys_wheel_timer 消费数据 ps: currQueueOffsetcheckpoint 读取出来的
  • 将消费出来的数据, 封装成 TimerRequest 投入到 enqueuePutQueue
  • currQueueOffset + 1 进入下一个循环 消费下一个 offset 节点

深入 TimerMessageStore 之 TimerEnqueuePutService

  • 消费 enqueuePutQueue 中的数据
  • shouldRunningDequeue && req.getDelayTime() < currWriteTimeMs 检查消费的消息是否已到达投递时间。
    • 到达时间。投递到 dequeuePutQueue.put(req);
    • 消息未到达时间 doEnqueue ->
      • timerWheel.getSlot(delayedTime) 获取延迟时间插槽。
      • 构建 ByteBuffer 投入 timerLog 中数据结构为:
      • |消息大小|前一个节点的pos|magic|log写入时间|延迟时间|offsetPy|sizePy|realTopic|0
      • timerLog.append 返回插入位置 ret
      • 构建 timerWheel |消息到达时间戳|firstPos|ret (timerLog.append返回位置)| 消息数量| 0|

深入 TimerMessageStore 之 TimerDequeueGetService

  • 消费 timerWheel 中的数据
  • 根据 currReadTimeMs 来获取 timerWheel 插槽数据
    • currReadTimeMs 初始化的时候 timerCheckpoint.getLastReadTimeMs() 读取的是上次最后消费的数据
    • 假设broker 宕机了一段时间。那么 currReadTimeMs 会按照上一次宕机的时间开始搜寻数据, 这样子宕机消息也不会丢失。会在启动的那段时间被投递出去
    • currReadTimeMsmoveReadTime 方法中会自增
  • timerWheel.getSlot(currReadTimeMs); 读取插槽数据
    • long currOffsetPy = slot.lastPos; 读取插槽属性, 最后一个pos节点
    • timerLog.getWholeBuffer(currOffsetPy) 根据 currOffsetPy 获取 SelectMappedBufferResult
    • timerLogSelectMappedBufferResult 中获取数据。
      • prevPos 上一个节点数据
      • enqueueTime 放入 timerLog 的时间
      • delayedTime 消息到达时间戳
      • offsetPy commitLog的数据位置
      • sizePy commitLog的数据大小
    • 构建 TimerRequest 讲消息投递到 dequeueGetQueue
    • currOffsetPy = prevPos 将位置移动到前一个,进行遍历

深入 TimerMessageStore 之 TimerDequeueGetMessageService

  • 默认有三个 TimerDequeueGetMessageService 实例同时消费 dequeueGetQueue
  • getMessageByCommitOffsetcommitLog 中读取原投递的消息数据
  • 读取 uniqkey 判断不在 deleteList 中的时候 将消息投递到 dequeuePutQueue 中去

深入 TimerMessageStore 之 TimerDequeuePutMessageService

  • 默认有三个 TimerDequeuePutMessageService 实例同时消费 dequeuePutQueue
  • convert(tr.getMsg(), tr.getEnqueueTime(), needRoll(tr.getMagic())); 将消息转换成原始的 topic 消息,清除无用属性
  • doPut -> messageStore.putMessage(message) 将消息投递到指定 messageQueue

TimerFlushService

  • timerLog 刷盘
  • timerWheel 刷盘
  • timerCheckpoint 刷盘

TimerMessageStore 初始化加载源码

  • timerLog.load() 加载文件
  • timerMetrics.load 加载文件
  • recover ->
    • recoverAndRevise(lastFlushPos, true) ps: (用于 timerWheltimerLog 的数据保持一致刷新)

      • lastFlushPos 最后一次刷盘的位置, 其实最终是拿到 timerlog -> mappedFile 的第几个文件
      • 遍历这个 mappedFile 的数据
      • timerWheel.reviseSlot 修改插槽数据。 检查这个时间的插槽是否已经有填充数据。
        • 如果有的话,刷新 lastPos (顺序遍历。这里最终还是会是最后一个 lastPos)
        • 如果不存在插槽数据 则插入插槽数据 putSlot
    • reviseQueueOffset(processOffset); 读取 timerLog 最后一个数据, 为了校验最后一个数据是否正常,是否能读取到消息。
    • 确认 currQueueOffset 数据
    • 确认 currReadTimeMs 数据

RocketMq5.0 任意延迟时间 TimerMessageStore 源码解析的更多相关文章

  1. abp vnext2.0核心组件之.Net Core默认DI组件切换到AutoFac源码解析

    老版Abp对Castle的严重依赖在vnext中已经得到了解决,vnext中DI容器可以任意更换,为了实现这个功能,底层架构相较于老版abp,可以说是进行了高度重构.当然这得益于.Net Core的D ...

  2. EventBus3.0源码解析

    本文主要介绍EventBus3.0的源码 EventBus是一个Android事件发布/订阅框架,通过解耦发布者和订阅者简化 Android 事件传递. EventBus使用简单,并将事件发布和订阅充 ...

  3. solr&lucene3.6.0源码解析(四)

    本文要描述的是solr的查询插件,该查询插件目的用于生成Lucene的查询Query,类似于查询条件表达式,与solr查询插件相关UML类图如下: 如果我们强行将上面的类图纳入某种设计模式语言的话,本 ...

  4. solr&lucene3.6.0源码解析(三)

    solr索引操作(包括新增 更新 删除 提交 合并等)相关UML图如下 从上面的类图我们可以发现,其中体现了工厂方法模式及责任链模式的运用 UpdateRequestProcessor相当于责任链模式 ...

  5. Heritrix 3.1.0 源码解析(三十七)

    今天有兴趣重新看了一下heritrix3.1.0系统里面的线程池源码,heritrix系统没有采用java的cocurrency包里面的并发框架,而是采用了线程组ThreadGroup类来实现线程池的 ...

  6. Masonry1.0.2 源码解析

    在了解Masonry框架之前,有必要先了解一下自动布局的概念.在iOS6之前,UI布局的方式是通过frame属性和Autoresizing来完成的,而在iOS6之后,苹果公司推出了AutoLayout ...

  7. Retrofit2.0源码解析

    欢迎访问我的个人博客 ,原文链接:http://wensibo.net/2017/09/05/retrofit/ ,未经允许不得转载! 今天是九月的第四天了,学校也正式开学,趁着大学最后一年的这大好时 ...

  8. 04、NetCore2.0下Web应用之Startup源码解析

    04.NetCore2.0Web应用之Startup源码解析   通过分析Asp.Net Core 2.0的Startup部分源码,来理解插件框架的运行机制,以及掌握Startup注册的最优姿势. - ...

  9. Android事件总线(二)EventBus3.0源码解析

    1.构造函数 当我们要调用EventBus的功能时,比如注册或者发送事件,总会调用EventBus.getDefault()来获取EventBus实例: public static EventBus ...

  10. 简单理解 OAuth 2.0 及资料收集,IdentityServer4 部分源码解析

    简单理解 OAuth 2.0 及资料收集,IdentityServer4 部分源码解析 虽然经常用 OAuth 2.0,但是原理却不曾了解,印象里觉得很简单,请求跳来跳去,今天看完相关介绍,就来捋一捋 ...

随机推荐

  1. 弱语言返回的数值型变量有可能是int,也有可能是string,该如何赋值给结构体

    包地址 github.com/jefferyjob/go-easy-util... 介绍 在解析弱语言类型返回的 Json 数据时,我们可能会遇到一些麻烦,比如 Json 数据中的数值型变量既可能是 ...

  2. 颜值即正义,献礼就业季,打造多颜色多字体双飞翼布局技术简历模版(Resume)

    一年好景君须记,最是橙黄橘绿时.金三银四,秣马厉兵,没有一个好看的简历模板怎么行?无论是网上随便下载还是花钱买,都是一律千篇的老式模版,平平无奇,味同嚼蜡,没错,蜡都要沿着嘴角流下来了.本次我们基于H ...

  3. c++基本数据结构

    基本数据结构: 一.线性表 1.顺序结构 线性表可以用普通的一维数组存储. 你可以让线性表可以完成以下操作(代码实现很简单,这里不再赘述): 返回元素个数. 判断线性表是否为空. 得到位置为p的元素. ...

  4. TF-IDF定义及实现

    TF-IDF定义及实现 定义 ​ TF-IDF的英文全称是:Term Frequency - Inverse Document Frequency,中文名称词频-逆文档频率,常用于文本挖掘,资讯检索等 ...

  5. 特性介绍 | MySQL 测试框架 MTR 系列教程(一):入门篇

    作者:卢文双 资深数据库内核研发 去年年底通过微信公众号[数据库内核]设定了一个目标--2023 年要写一系列 特性介绍+内核解析 的文章(现阶段还是以 MySQL 为主). 虽然关注者很少,但本着& ...

  6. Go语言实现文件服务器

    主调函数,设置路由表 package main import ( "fmt" "net/http" "store/handler" ) fu ...

  7. 33-webpack详细配置output

    const { resolve } = require('path') const HtmlWebpackPlugin = require('html-webpack-plugin') module. ...

  8. X配置文件xorg.conf分析

    X配置文件xorg.conf分析 转载于:http://blog.csdn.NET/comcat/archive/2007/04/02/1549658.aspx 作者:壮志凌云的csdn博客 X的配置 ...

  9. ts中接口

    前言:ts定义接口的任意一个属性 interface IPerson { name: string age: number family?: any[] // Error,因为不是任意类型的子集 [p ...

  10. 云原生时代崛起的编程语言Go基础实战

    @ 目录 概述 定义 使用场景 Go 安全 使用须知 搜索工具 Go基础命令 标准库 基础语法 Effective Go 概览 命名规范 注释 变量 常量(const) 控制结构 数据类型 迭代(ra ...