本文介绍了DefaultMQPushConsumerImpl消费者,客户端负载均衡相关知识点。本文从DefaultMQPushConsumerImpl启动过程到实现负载均衡,从源代码一步一步分析,共分为6个部分进行介绍,其中第6个部分 rebalanceByTopic 为负载均衡的核心逻辑模块,具体过程运用了图文进行阐述。
 
介绍之前首先抛出几个问题:
1. 要做负载均衡,首先要解决的一个问题是什么?
2. 负载均衡是Client端处理还是Broker端处理?
个人理解:
1. 要做负载均衡,首先要做的就是信号收集。
所谓信号收集,就是得知道每一个consumerGroup有哪些consumer,对应的topic是谁。信号收集分为Client端信号收集与Broker端信号收集两个部分。
2. 负载均衡放在Client端处理。
具体做法是:消费者客户端在启动时完善rebalanceImpl实例,同时拷贝订阅信息存放rebalanceImpl实例对象中,另外也是很重要的一个步骤 -- 通过心跳消息,不停的上报自己到所有Broker,注册RegisterConsumer,等待上述过程准备好之后在Client端不断执行的负载均衡服务线程从Broker端获取一份全局信息(该consumerGroup下所有的消费Client),然后分配这些全局信息,获取当前客户端分配到的消费队列。
 
本文具体的内容:
I. copySubscription
Client端信号收集,拷贝订阅信息。

在DefaultMQPushConsumerImpl.start()时,会将消费者的topic订阅关系设置到rebalanceImpl的SubscriptionInner的map中用于负载:

  1. private void copySubscription() throws MQClientException {
  2. try {
  3. //注:一个consumer对象可以订阅多个topic
  4. Map<String, String> sub = this.defaultMQPushConsumer.getSubscription();
  5. if (sub != null) {
  6. for (final Map.Entry<String, String> entry : sub.entrySet()) {
  7. final String topic = entry.getKey();
  8. final String subString = entry.getValue();
  9. SubscriptionData subscriptionData =
  10. FilterAPI.buildSubscriptionData(this.defaultMQPushConsumer.getConsumerGroup(),//
  11. topic, subString);
  12. this.rebalanceImpl.getSubscriptionInner().put(topic, subscriptionData);
  13. }
  14. }
  15.  
  16. if (null == this.messageListenerInner) {
  17. this.messageListenerInner = this.defaultMQPushConsumer.getMessageListener();
  18. }
  19.  
  20. switch (this.defaultMQPushConsumer.getMessageModel()) {
  21. case BROADCASTING:
  22. break;
  23. case CLUSTERING:
  24. final String retryTopic = MixAll.getRetryTopic(this.defaultMQPushConsumer.getConsumerGroup());
  25. SubscriptionData subscriptionData =
  26. FilterAPI.buildSubscriptionData(this.defaultMQPushConsumer.getConsumerGroup(),//
  27. retryTopic, SubscriptionData.SUB_ALL);
  28. this.rebalanceImpl.getSubscriptionInner().put(retryTopic, subscriptionData);
  29. break;
  30. default:
  31. break;
  32. }
  33. }
  34. catch (Exception e) {
  35. throw new MQClientException("subscription exception", e);
  36. }
  37. }
FilterAPI.buildSubscriptionData接口将订阅关系转换为SubscriptionData 数据,其中subString包含订阅tag等信息。另外,如果该消费者的消费模式为集群消费,则会将retry的topic一并放到。
 
II. 完善rebalanceImpl实例
Client继续收集信息:
  1. this.rebalanceImpl.setConsumerGroup(this.defaultMQPushConsumer.getConsumerGroup());
  2. this.rebalanceImpl.setMessageModel(this.defaultMQPushConsumer.getMessageModel());
  3. this.rebalanceImpl.setAllocateMessageQueueStrategy(this.defaultMQPushConsumer
  4. .getAllocateMessageQueueStrategy());
  5. this.rebalanceImpl.setmQClientFactory(this.mQClientFactory);
本文以DefaultMQPushConsumerImpl为例,因此this对象类型为DefaultMQPushConsumerImp。
 
III. this.rebalanceService.start()
开启负载均衡服务。this.rebalanceService是一个RebalanceService实例对象,它继承与ServiceThread,是一个线程类。 this.rebalanceService.start()执行时,也即执行RebalanceService线程体:
  1. @Override
  2. public void run() {
  3. log.info(this.getServiceName() + " service started");
  4.  
  5. while (!this.isStoped()) {
  6. this.waitForRunning(WaitInterval);
  7. this.mqClientFactory.doRebalance();
  8. }
  9.  
  10. log.info(this.getServiceName() + " service end");
  11. }
IV. this.mqClientFactory.doRebalance
客户端遍历消费组table,对该客户端上所有消费者独立进行负载均衡,分发消费队列:
  1. public void doRebalance() {
  2. for (String group : this.consumerTable.keySet()) {
  3. MQConsumerInner impl = this.consumerTable.get(group);
  4. if (impl != null) {
  5. try {
  6. impl.doRebalance();
  7. } catch (Exception e) {
  8. log.error("doRebalance exception", e);
  9. }
  10. }
  11. }
  12. }
V. MQConsumerInner.doRebalance
由于本文以DefaultMQPushConsumerImpl消费过程为例,即DefaultMQPushConsumerImpl.doRebalance:
  1. @Override
  2. public void doRebalance() {
  3. if (this.rebalanceImpl != null) {
  4. this.rebalanceImpl.doRebalance();
  5. }
  6. }
步骤II 中完善了rebalanceImpl实例,为调用rebalanceImpl.doRebalance()提供了初始数据。
rebalanceImpl.doRebalance()过程如下:
  1. public void doRebalance() {
  2.      // 前文copySubscription中初始化了SubscriptionInner
  3. Map<String, SubscriptionData> subTable = this.getSubscriptionInner();
  4. if (subTable != null) {
  5. for (final Map.Entry<String, SubscriptionData> entry : subTable.entrySet()) {
  6. final String topic = entry.getKey();
  7. try {
  8. this.rebalanceByTopic(topic);
  9. } catch (Exception e) {
  10. if (!topic.startsWith(MixAll.RETRY_GROUP_TOPIC_PREFIX)) {
  11. log.warn("rebalanceByTopic Exception", e);
  12. }
  13. }
  14. }
  15. }
  16.  
  17. this.truncateMessageQueueNotMyTopic();
  18. }

VI. rebalanceByTopic -- 核心步骤之一
rebalanceByTopic方法中根据消费者的消费类型为BROADCASTING或CLUSTERING做不同的逻辑处理。CLUSTERING逻辑包括BROADCASTING逻辑,本部分只介绍集群消费负载均衡的逻辑。
集群消费负载均衡逻辑主要代码如下(省略了log等代码):

  1. //1.从topicSubscribeInfoTable列表中获取与该topic相关的所有消息队列
  2. Set<MessageQueue> mqSet = this.topicSubscribeInfoTable.get(topic);
  3. //2. 从broker端获取消费该消费组的所有客户端clientId
  4. List<String> cidAll = this.mQClientFactory.findConsumerIdList(topic, consumerGroup);
  5. f (null == mqSet) { ... }
  6. if (null == cidAll) { ... }
  7. if (mqSet != null && cidAll != null) {
  8. List<MessageQueue> mqAll = new ArrayList<MessageQueue>();
  9. mqAll.addAll(mqSet);
  10. Collections.sort(mqAll);
  11. Collections.sort(cidAll);
  12.  
  13.      // 3.创建DefaultMQPushConsumer对象时默认设置为AllocateMessageQueueAveragely
  14. AllocateMessageQueueStrategy strategy = this.allocateMessageQueueStrategy;
  15.  
  16. List<MessageQueue> allocateResult = null;
  17. try {
  18.          // 4.调用AllocateMessageQueueAveragely.allocate方法,获取当前client分配消费队列
  19. allocateResult = strategy.allocate(
  20. this.consumerGroup,
  21. this.mQClientFactory.getClientId(),
  22. mqAll,
  23. cidAll);
  24. } catch (Throwable e) {
  25. return;
  26. }
  27.     // 5. 将分配得到的allocateResult 中的队列放入allocateResultSet 集合
  28. Set<MessageQueue> allocateResultSet = new HashSet<MessageQueue>();
  29. if (allocateResult != null) {
  30. allocateResultSet.addAll(allocateResult);
  31. }

  32.      //6. 更新updateProcessQueue
  33. boolean changed = this.updateProcessQueueTableInRebalance(topic, allocateResultSet);
  34. if (changed) {
  35. this.messageQueueChanged(topic, mqSet, allocateResultSet);
  36. }
  37. }
注:BROADCASTING逻辑只包含上述的1、6。
集群消费负载均衡逻辑中的1、2、4这三个点相关知识为其核心过程,各个点相关知识如下:
 
第1点:从topicSubscribeInfoTable列表中获取与该topic相关的所有消息队列

第2点: 从broker端获取消费该消费组的所有客户端clientId
首先,消费者对象不断地向所有broker发送心跳包,上报自己,注册并更新订阅关系以及客户端ChannelInfoTable;之后,客户端在做消费负载均衡时获取那些消费客户端,对这些客户端进行负载均衡,分发消费的队列。具体过程如下图所示:

第4点:调用AllocateMessageQueueAveragely.allocate方法,获取当前client分配消费队列
 

注:上图中cId1、cId2、...、cIdN通过 getConsumerIdListByGroup 获取,它们在这个ConsumerGroup下所有在线客户端列表中。
当前消费对进行负载均衡策略后获取对应的消息消费队列。具体的算法很简单,可以看源码。
 
 
 

rocketmq消费负载均衡--push消费为例的更多相关文章

  1. RocketMQ-2.RocketMQ的负载均衡

    目录 RocketMQ的负载均衡 producer对MessageQueue的负载均衡 producer负载均衡 系统计算路由MessageQueue 自定义路由MessageQueue Consum ...

  2. Nginx 简单的负载均衡配置演示样例

    近期在做开放查询应用的时候,因为数据两天特别多,两千多万条呢,用户訪问需求也比較大,所以就用nginx做了 负载均衡,以下是改动之后的相关内容. http://www.cnblogs.com/xiao ...

  3. RocketMQ(消息重发、重复消费、事务、消息模式)

    分布式开放消息系统(RocketMQ)的原理与实践 RocketMQ基础:https://github.com/apache/rocketmq/tree/rocketmq-all-4.5.1/docs ...

  4. 深入剖析 RocketMQ 源码 - 负载均衡机制

    RocketMQ作为一款流行的消息中间件在各大互联网应用广泛,本文主要分析RocketMq在消息生产和消费过程中的负载均衡机制,并创新提出消费端负载均衡策略的改写以实现固定IP消费的可能.

  5. 通过http、https域名访问静态网页、nginx配置负载均衡(nginx配置)

    很多场景下需要可以通过浏览器访问静态网页,不想把服务器ip地址直接暴露出来,通过nginx可以解决这个问题. 实现http域名访问静态网页 1.域名解析配置(本文都是以阿里云为例,其他平台,操作步骤类 ...

  6. ③SpringCloud 实战:使用 Ribbon 客户端负载均衡

    这是SpringCloud实战系列中第三篇文章,了解前面第两篇文章更有助于更好理解本文内容: ①SpringCloud 实战:引入Eureka组件,完善服务治理 ②SpringCloud 实战:引入F ...

  7. 玩转Spring Cloud之服务注册发现(eureka)及负载均衡消费(ribbon、feign)

    如果说用Spring Boot+Spring MVC是开发单体应用(或单体服务)的利器,那么Spring Boot+Spring MVC+Spring Cloud将是开发分布式应用(快速构建微服务)的 ...

  8. spring cloud(服务消费者(利用ribbon实现服务消费及负载均衡)——初学二)

    Ribbon是一个基于HTTP和TCP客户端的负载均衡器,利用ribbon实现服务消费,并实现客户端的负载均衡. 一.准备工作(利用上一节的内容) 启动服务注册中心 启动computer-servic ...

  9. Spring Cloud ---- 服务消费与负载均衡(feign)

    feign是一个声明式的伪客户端,只需要创建一个接口并且注解,它具有可插拔的特性.feign集合了Ribbon,再与Eurake结合实现服务的注册发现与负载均衡.结合Hystrix,具有熔断功能. 1 ...

随机推荐

  1. odoo10 修改产品单价的小数点位数

    odoo10 修改产品单价的小数点位数 由于产品价格原因,单价需要保留小数点后 5 位,所以需要修改单价的小数点位数. 开启开发模式 找到数据库编辑 找到小数点精度 修改产品的小数点位数

  2. [LeetCode系列]有序链表转换为平衡BST的递归解法

    给定有序链表(元素由小到大), 试问如何将其转换为一个平衡BST? 平衡BST: 任意节点的左右子树的深度差值不大于1. 主要思想是用递归. Trick是使用快慢指针来获取中间节点. 获得中间节点后, ...

  3. 1.远程仓库的使用(github)

    1.登录Github,新建一个仓库(远程仓库) (1)使用Github账号密码登录 (2)点击+旁边的小三角,选择new repository--输入repository name--点击create ...

  4. SQL的复习与总结

    检索数据 关键字: SELECT DISTINCT LIMIT OFFSET FROM SELECT与FROM用于基础的检索,基本语法为: SELECT column_name,column_name ...

  5. Markdown初步使用

    一.兼容 HTML Markdown 的理念是,能让文档更容易读.写和随意改.HTML 是一种发布的格式,Markdown 是一种书写的格式.就这样,Markdown 的格式语法只涵盖纯文本可以涵盖的 ...

  6. 笔记:LNK2001不代表链接器真的需要链接相关符号

    环境:VS2008   我们都知道,链接器在生成可执行程序时,会忽略那些没有用到的符号.但是昨天遇到一个链接问题,看起来与这条基本策略并不相符.首先看一个静态链接库的结构:   lib | |---- ...

  7. Zookeeper--集群管理

    Zookeeper--集群管理 在多台服务器组成的集群中,需要监控每台服务器的状态,一旦某台服务器挂掉了或有新的机器加入集群,集群都要感知到,从而采取相应的措施.一个主动的集群可以自动感知节点的死亡和 ...

  8. python is 和 == 的区别

    一.is 和 == 的区别 == 比较 比较的俩边的值 is 比较 比较的是内存地址 id() 二.小数据池 数字小数据池的范围 -5 ~ 256 字符串中如果有特殊字符他们的内存地址就不一样 字符串 ...

  9. String.Format数字格式化输出 {0:N2} {0:D2} {0:C2} (转)

    String.Format数字格式化输出 {:N2} {:D2} {:C2} (转) //格式为sring输出 // Label1.Text = string.Format("asdfads ...

  10. shelve模块(超级好用~!)

    ''' python中的shelve模块,可以提供一些简单的数据操作 他和python中的dbm很相似. 区别如下: 都是以键值对的形式保存数据,不过在shelve模块中, key必须为字符串,而值可 ...