草庐IT

RocketMQ消息发送流程

Leo聊技术 2023-03-28 原文
大家好,我是Leo。

今天聊一下RocketMQ消息发送,重试机制,故障延迟机制,获取路由机制,消息队列的选择

消息发送

关系图

首先放一下Broker Cluster,Broker,Topic,Queue的关系图。因为下文主要会沿着这四块进行梳理

发送的三种方式

消息发送的三种方式

  • 同步:发送者向MQ发送一条消息后,一直等待服务器返回成功才继续下一个。
  • 异步:发送者向MQ发送一条消息后,通过回调函数调用消息发布函数继续发送,主线程立即返回。
  • 单向:发送者向MQ发送一条消息后,直接返回,不等待消息服务器的结果,也不注册函数,简单来说,就是只管发。其他啥也不管。
源码

package org.apache.rocketmq.client.impl;

/**
* 消息发送的三种方式
*/
public enum CommunicationMode {
// 同步发送
SYNC,
// 异步发送
ASYNC,
// 单向发送
ONEWAY,
}

重试机制

RocketMQ的重试机制,主要由下列两个参数决定。默认重试次数为2次,重试机制提高了消息发送成功的几率。

/**
* 同步模式下内部尝试发送消息的最大次数
*/
private int retryTimesWhenSendFailed = 2
/**
* 异步模式下内部尝试发送消息的最大次数
*/
private int retryTimesWhenSendAsyncFailed = 2;

故障延迟机制

RocketMQ的故障延迟机制,主要由下列参数决定,默认是不开启的。故障延迟机制,主要体现在集群的时候,当broker发送错误时,可以有效的规避多次发送消息都发往一个broker(queue)的错误。

/**
* 默认不启用Broker故障延迟机制。
*/
private boolean sendLatencyFaultEnable = false;

获取路由信息机制

消息在发送时,需要知道,要发往哪个broker。首先会去 ​​brokerAddrTable​​ 中查找当前brokerName是否存在在本地的缓存中

  1. 如果成功返回brokerName
  2. 否则就返回null
/* Broker Name */ /* brokerId */ /* address */
private final ConcurrentMap<String, HashMap<Long, String>> brokerAddrTable;
如果成功一帆风顺,如果找不到的话,肯定要做一些安全处理。

如果找不到的话,会通过 ​​tryToFindTopicPublishInfo​​ 函数尝试查找主题发布信息

  1. topicPublishInfoTable 缓存中根据topic名称查找是否存在
  2. 如果没有缓存,会创建一个以topic名称为key,空 TopicPublishInfo 为value到 topicPublishInfoTable ,然后更新到 NameServer
  3. 如果消息的路由信息存在,并且 MessageQueue 不为空 直接返回路由信息
  4. 否则使用默认主题
/* topic */
private final ConcurrentMap<String, TopicPublishInfo> topicPublishInfoTable

消息队列的选择

开启故障延迟

遍历主题队列的消息队列,根据访问次数进行随机自增取模。

如果当前消息队列是可用的就直接返回。 函数名 ​​isAvailable​

如果是不可用的,从失败的brokeName列表中通过 ​​pickOneAtLeast​​ 函数选择一个可用的broker。拿到brokerName之后,再根据brokerName反查这个队列的写队列数

  1. 如果小于0说明该broker依据恢复,从失败的条目中移出当前broker
  2. 如果大于0通过 selectOneMessageQueue 函数选出一个消息队列
/**
* 失败的broker列表
*/
private final LatencyFaultTolerance < String > latencyFaultTolerance;
没有开启故障延迟

如果上一次选择的执行发送消息失败的broker名称为空,它会通过 ​​selectOneMessageQueue()​​ 函数对当前访问的次数取绝对值,然后与消息队列的大小取模得到一个下标,然后从 ​​messageQueueList​​ 中根据下标取出 ​​MessageQueue​

如果上一次选择的执行发送消息失败的broker名称不为空,会遍历消息队列,对当前访问的次数取绝对值,然后与消息队列的大小取模得到一个下标后,拿着下标获取对应的 ​​brokerName​​ 并且判断当前的 ​​brokerName​​ 是否与上一次发送消息失败的 ​​brokerName​​ 相等,

  1. 如果相等就遍历所有主题内的消息队列。假如还是没有找到一个合适的,就会随机选择一个
  2. 如果不相等,就把当前下标的 MessageQueue 返出去。
下图的json字符串就是 ​​MessageQueue​​ 信息

/**
* 该主题队列的消息队列
*/
private List < MessageQueue > messageQueueList = new ArrayList < MessageQueue > ();

[
{
"brokerName": "broker-a",
"queueId": 0
},
{
"brokerName": "broker-a",
"queueId": 1
}, {
"brokerName": "broker-b",
"queueId": 0
}, {
"brokerName": "broker-b",
"queueId": 1
}, {
"brokerName": "broker-c",
"queueId": 0
}, {
"brokerName": "broker-c",
"queueId": 1
}
]
开启故障延迟机制中的可用依据是:检查时间是否到达了下次可使用的时间点

如果没有该机制,如果broker宕机,由于路由算法中的消息队列是按broker排序的,顺序选择,如果上一次根据路由算法选择的是宕机的broker的第一个队列,那么随后的下次选择的是宕机broker的第二个队列,消息发送很有可能会失败,再次引发重试,带来不必要的性能损耗。

selectOneMessageQueue()也可以看成是兜底策略-轮询算法

同步发送

由上文得知,消息发送有三种方式。我们先看一下同步发送主要做了哪些事情。

DefaultMQProducerImpl的send函数是发送消息的入口

  1. 通过 makeSureStateOK 函数检查服务状态是否正常
  2. 通过 checkMessage 函数校验 MessageDefaultMQProducer 是否符合发送的规则
  3. 校验消息的主题不能等于消息队列集合的主题信息以及以上操作是否超时
  4. 校验brokerName是否存在,如果不存在通过 findBrokerAddressInPublish 函数去nameserver拉取
  5. 通过 brokerVIPChannel 函数校验是否使用了vip管道,如果使用了管道在原来的基础上把 端口-2
  6. 通过配置信息获取生成uniqId的算法规则以及封装 Message 的实例信息
  7. Message 的 body 信息进行压缩
  8. 获取当前的配置信息,是否启用事务。
  9. 封装发送消息模板权限信息 SendMessageContext,构造请求头
  10. 发送之前,校验一下 Topid 类型是否属于重试类型消息(这里可以看看下列注释)
  11. 通过 CommunicationMode 枚举类型判断当前是什么发送方式
  12. 判断当前是正常指令发送,还是RPC指令发送,判断是否对字段进行压缩处理(简化压缩有助于提速序列化速度)
  13. 根据broker地址获取Netty对应的Channel,并远程调用(这里的发送,用的是Netty框架)
  14. 通过 processSendResponse 函数处理同步返回的参数,如果参数为0,说明发送成功。最后封装 SendResult 返回
第四步中,如果在nameserver拉取不到,说明服务宕机了。

第五步中,vip的管道配置从配置文件中的com.rocketmq.sendMessageWithVIPChannel得知

第六步中,批量信息不支持压缩

第十步中,如果是重试消息,通过获取自定义重试次数,在请求头区分特别处理

第十一步中,因为这里介绍的是同步发送,就只写同步发送流程了,异步,单向会在下面段落体现出来

第十二步中,通过配置文件中org.apache.rocketmq.client.sendSmartMsg得知字段是否简化压缩

异步发送

聊完同步发送,我们看一下异步发送

DefaultMQProducerImpl的send函数是发送消息的入口(这里跟同步的区别是多了一个 ​​SendCallback​​)

  1. 在生产者生产消息发送时,通过 ExecutorService 新增一个异步任务进行发送(可看下列注释,可看源码区)
  2. 通过 makeSureStateOK 函数检查服务状态是否正常
  3. 通过 checkMessage 函数校验 MessageDefaultMQProducer 是否符合发送的规则
  4. 校验消息的主题不能等于消息队列集合的主题信息以及以上操作是否超时
  5. 校验brokerName是否存在,如果不存在通过 findBrokerAddressInPublish 函数去nameserver拉取
  6. 通过 brokerVIPChannel 函数校验是否使用了vip管道,如果使用了管道在原来的基础上把 端口-2
  7. 通过配置信息获取生成uniqId的算法规则以及封装 Message 的实例信息
  8. Message 的 body 信息进行压缩
  9. 获取当前的配置信息,是否启用事务。
  10. 封装发送消息模板权限信息 SendMessageContext,构造请求头
  11. 发送之前,校验一下 Topid 类型是否属于重试类型消息(这里可以看看下列注释)
  12. 通过 CommunicationMode 枚举类型判断当前是什么发送方式
  13. 判断当前是正常指令发送,还是RPC指令发送,判断是否对字段进行压缩处理(简化压缩有助于提速序列化速度)
  14. 根据broker地址获取Netty对应的Channel,并远程调用(这里的发送,用的是Netty框架)
  15. 通过 processSendResponse 函数处理并且利用委托 remotingClient.invokeAsync 等待返回的SendResult 结构体
  16. 上一步骤再插一句,异步发送会处理一个 updateFaultItem 函数记录当前 不可以时间/可用时间 时间
第一步中 借助 java.util.concurrent.ExecutorService ,实现一个线程池达到可以让任务在后台执行。

第十五步中 RemotingClient的invokeAsync函数

单向发送

单向发送,与同步发送相似。与同步发送不同的是通过 RemotingClient#invokeOneway 函数委托发送。

从 invokeOnway进入后

  1. 如果当前addr为空,获取和创建Nameserver通道
  2. 创建成功后,只要通道是活跃的,并且不为空,就利用Netty框架进行 writeAndFlush
创建通道时,通过 ReentrantLock 对nameSeverChannel加锁,超时时长为3秒

源码

DefaultMQProducerImpl#send同步函数

/**
* 内核同步发送
* @param msg
* @param mq
* @return
* @throws MQClientException
* @throws RemotingException
* @throws MQBrokerException
* @throws InterruptedException
*/
public SendResult send(Message msg, MessageQueue mq) throws MQClientException, RemotingException, MQBrokerException, InterruptedException {
return send(msg, mq, this.defaultMQProducer.getSendMsgTimeout());
}

/**
* 内核同步发送下的 send子函数
* @param msg
* @param mq
* @param timeout
* @return
* @throws MQClientException
* @throws RemotingException
* @throws MQBrokerException
* @throws InterruptedException
*/
public SendResult send(Message msg, MessageQueue mq, long timeout) throws MQClientException, RemotingException, MQBrokerException, InterruptedException {
long beginStartTime = System.currentTimeMillis();
this.makeSureStateOK();
Validators.checkMessage(msg, this.defaultMQProducer);

if (!msg.getTopic().equals(mq.getTopic())) {
// 消息的主题不等于mq的主题
throw new MQClientException("message's topic not equal mq's topic", null);
}

long costTime = System.currentTimeMillis() - beginStartTime;
if (timeout < costTime) {
throw new RemotingTooMuchRequestException("call timeout");
}

return this.sendKernelImpl(msg, mq, CommunicationMode.SYNC, null, null, timeout);
}
DefaultMQProducerImpl#send异步函数

/**
* 内核异步
*
* @param msg
* @param mq
* @param sendCallback
* @throws MQClientException
* @throws RemotingException
* @throws InterruptedException
*/
public void send(Message msg, MessageQueue mq, SendCallback sendCallback) throws MQClientException, RemotingException, InterruptedException {
send(msg, mq, sendCallback, this.defaultMQProducer.getSendMsgTimeout());
}
/**
* 内核异步发送下的 send 子函数

* @param msg
* @param mq
* @param sendCallback
* @param timeout the <code>sendCallback</code> will be invoked at most time
* @throws MQClientException
* @throws RemotingException
* @throws InterruptedException
*/
@Deprecated
public void send(final Message msg, final MessageQueue mq, final SendCallback sendCallback, final long timeout) throws MQClientException, RemotingException, InterruptedException {
final long beginStartTime = System.currentTimeMillis();
ExecutorService executor = this.getAsyncSenderExecutor();
try {
executor.submit(new Runnable() {
@Override
public void run() {
try {
makeSureStateOK();
Validators.checkMessage(msg, defaultMQProducer);

if (!msg.getTopic().equals(mq.getTopic())) {
throw new MQClientException("message's topic not equal mq's topic", null);
}
long costTime = System.currentTimeMillis() - beginStartTime;
if (timeout > costTime) {
try {
sendKernelImpl(msg, mq, CommunicationMode.ASYNC, sendCallback, null, timeout - costTime);
} catch (MQBrokerException e) {
throw new MQClientException("unknown exception", e);
}
} else {
sendCallback.onException(new RemotingTooMuchRequestException("call timeout"));
}
} catch (Exception e) {
sendCallback.onException(e);
}

}

});
} catch (RejectedExecutionException e) {
throw new MQClientException("executor rejected ", e);
}

}
DefaultMQProducerImpl#sendOneway函数

/**
* 内核单向发送
*/
public void sendOneway(Message msg, MessageQueue mq) throws MQClientException, RemotingException, InterruptedException {
this.makeSureStateOK();
Validators.checkMessage(msg, this.defaultMQProducer);

try {
this.sendKernelImpl(msg, mq, CommunicationMode.ONEWAY, null, null, this.defaultMQProducer.getSendMsgTimeout());
} catch (MQBrokerException e) {
throw new MQClientException("unknown exception", e);
}
}

往期推荐

​2022年4月文章目录整理​

​RocketMQ数据压缩的那套把戏​

​图文并茂!深入理解RocketMQ的刷盘机制​

​图文并茂!深入了解RocketMQ的过期删除机制​

​图文并茂!深入了解RocketMQ的内存映射机制​

结尾

单向发送那里如果有问题,可以私信我。我们一起交流!

关于整篇的思路与总结。主要是从RocketMQ的消息发送入手的,消息发送主要分三种

  1. 同步
  2. 异步
  3. 单向
从三种方式各自深入源码进行分析得知,同步,单向,异步流程大致相同

异步发送与同步发送最大的不同: 异步发送在同步发送的基础上利用ExecutorService 进行初始化异步任务。在执行完成之后,还会有一个 ​​updateFaultItem​​ 时间记录处理。

  1. 正常情况下,会传一个false值,false代表没有问题,会采用我们自己计算的时间戳赋值
  2. 异常情况下,会传一个true值,true代表有问题,会采用默认时间30s赋值
单向又与同步,异步有些不同,单向因为不需要知道是否成功,所以他把这条发送请求进行委托处理(利用Netty框架Channel的 ​​writeAndFlush​​)

有关后端方面的问题,非常欢迎大家咨询我,我们在群内一起讨论! 我们下期再见!

欢迎『点赞』、『在看』、『转发』三连支持一下,下次见~


有关RocketMQ消息发送流程的更多相关文章

  1. ruby-on-rails - 如何在 Rails View 上显示错误消息? - 2

    我是rails的新手,想在form字段上应用验证。myviewsnew.html.erb.....模拟.rbclassSimulation{:in=>1..25,:message=>'Therowmustbebetween1and25'}end模拟Controller.rbclassSimulationsController我想检查模型类中row字段的整数范围,如果不在范围内则返回错误信息。我可以检查上面代码的范围,但无法返回错误消息提前致谢 最佳答案 关键是您使用的是模型表单,一种显示ActiveRecord模型实例属性的表单。c

  2. jquery - 我的 jquery AJAX POST 请求无需发送 Authenticity Token (Rails) - 2

    rails中是否有任何规定允许站点的所有AJAXPOST请求在没有authenticity_token的情况下通过?我有一个调用Controller方法的JqueryPOSTajax调用,但我没有在其中放置任何真实性代码,但调用成功。我的ApplicationController确实有'request_forgery_protection'并且我已经改变了config.action_controller.consider_all_requests_local在我的environments/development.rb中为false我还搜索了我的代码以确保我没有重载ajaxSend来发送

  3. ruby - 使用 Ruby 通过 Outlook 发送消息的最简单方法是什么? - 2

    我的工作要求我为某些测试自动生成电子邮件。我一直在四处寻找,但未能找到可以快速实现的合理解决方案。它需要在outlook而不是其他邮件服务器中,因为我们有一些奇怪的身份验证规则,我们需要保存草稿而不是仅仅发送邮件的选项。显然win32ole可以做到这一点,但我找不到任何相当简单的例子。 最佳答案 假设存储了Outlook凭据并且您设置为自动登录到Outlook,WIN32OLE可以很好地完成此操作:require'win32ole'outlook=WIN32OLE.new('Outlook.Application')message=

  4. Ruby - 如何将消息长度表示为 2 个二进制字节 - 2

    我正在使用Ruby,我正在与一个网络端点通信,该端点在发送消息本身之前需要格式化“header”。header中的第一个字段必须是消息长度,它被定义为网络字节顺序中的2二进制字节消息长度。比如我的消息长度是1024。如何将1024表示为二进制双字节? 最佳答案 Ruby(以及Perl和Python等)中字节整理的标准工具是pack和unpack。ruby的packisinArray.您的长度应该是两个字节长,并且按网络字节顺序排列,这听起来像是n格式说明符的工作:n|Integer|16-bitunsigned,network(bi

  5. ruby-on-rails - 在 Flash 警报 Rails 3 中显示错误消息 - 2

    如果我在模型中设置验证消息validates:name,:presence=>{:message=>'Thenamecantbeblank.'}我如何让该消息显示在闪光警报中,这是我迄今为止尝试过的方法defcreate@message=Message.new(params[:message])if@message.valid?ContactMailer.send_mail(@message).deliverredirect_to(root_path,:notice=>"Thanksforyourmessage,Iwillbeintouchsoon")elseflash[:error]

  6. ruby-on-rails - 在 RSpec 中,如何以任意顺序期望具有不同参数的多条消息? - 2

    RSpec似乎按顺序匹配方法接收的消息。我不确定如何使以下代码工作:allow(a).toreceive(:f)expect(a).toreceive(:f).with(2)a.f(1)a.f(2)a.f(3)我问的原因是a.f的一些调用是由我的代码的上层控制的,所以我不能对这些方法调用添加期望。 最佳答案 RSpecspy是测试这种情况的一种方式。要监视一个方法,用allowstub,除了方法名称之外没有任何约束,调用该方法,然后expect确切的方法调用。例如:allow(a).toreceive(:f)a.f(2)a.f(1)

  7. ruby - 是否可以在不实际发送或读取数据的情况下查明 ruby​​ 套接字是否处于 ESTABLISHED 或 CLOSE_WAIT 状态? - 2

    s=Socket.new(Socket::AF_INET,Socket::SOCK_STREAM,0)s.connect(Socket.pack_sockaddr_in('port','hostname'))ssl=OpenSSL::SSL::SSLSocket.new(s,sslcert)ssl.connect从这里开始,如果ssl连接和底层套接字仍然是ESTABLISHED,或者它是否在默认值7200之后进入CLOSE_WAIT,我想检查一个线程几秒钟甚至更糟的是在实际上不需要.write()或.read()的情况下关闭。是用select()、IO.select()还是其他方法完成

  8. ruby-on-rails - 闪存消息存储在哪里? - 2

    我以为它们存储在cookie中-但不,检查cookie没有任何结果。session也不存储它们。那么,我在哪里可以找到它们?我需要这个来直接设置它们(而不是通过flashhash)。 最佳答案 它们存储在inyoursessionstore.自rails2.0以来的默认设置是cookie存储,但请检查config/initializers/session_store.rb以检查您是否使用默认设置以外的东西。 关于ruby-on-rails-闪存消息存储在哪里?,我们在StackOverf

  9. Ruby SSL 错误 - sslv3 警报意外消息 - 2

    我正在尝试在ruby​​脚本中连接到服务器https://www.xpiron.com/schedule。但是,当我尝试连接时:require'open-uri'doc=open('https://www.xpiron.com/schedule')我收到以下错误消息:OpenSSL::SSL::SSLError:SSL_connectreturned=1errno=0state=SSLv2/v3readserverhelloA:sslv3alertunexpectedmessagefrom/usr/local/lib/ruby/1.9.1/net/http.rb:678:in`conn

  10. ruby-on-rails - Ruby on Rails - 需要在每周的特定时间将消息发送到电子邮件 - 2

    我想知道我应该如何着手这个项目。我需要每周向人们发送一次电子邮件。但是,这必须在每周的特定时间自动生成并发送。编码有多难?我需要知道是否有任何书籍可以提供帮助,或者你们中的任何人是否可以指导我。它必须使用ruby​​onrails进行编程。因此有一个网络服务和数据库集成。干杯 最佳答案 为什么这么复杂?您只需安排工作。您可以使用Delayed::Job例如。Delayed::Job让您可以使用run_at符号在特定时间安排作业,如下所示:Delayed::Job.enqueue(SendEmailJob.new(...),:run_

随机推荐