RocketMQ概述

RocketMQ 是一款分布式、队列模型的消息中间件,具有以下特点: 能够保证严格的消息顺序 提供丰富的消息拉取模式 高效的订阅者水平扩展能力 实时的消息订阅机制 亿级消息堆积能力

RocketMQ包含的组件

NameServer:单点,供Producer和Consumer获取Broker地址

Producer:产生并发送消息

Consumer:接受并消费消息

Broker:消息暂存,消息转发

Name Server

Name Server是RocketMQ的寻址服务。用于把Broker的路由信息做聚合。客户端依靠Name Server决定去获取对应topic的路由信息,从而决定对哪些Broker做连接。

Name Server是一个几乎无状态的结点,Name Server之间采取share-nothing的设计,互不通信。

对于一个Name Server集群列表,客户端连接Name Server的时候,只会选择随机连接一个结点,以做到负载均衡。

Name Server所有状态都从Broker上报而来,本身不存储任何状态,所有数据均在内存。

如果中途所有Name Server全都挂了,影响到路由信息的更新,不会影响和Broker的通信。

Broker

Broker是处理消息存储,转发等处理的服务器。

Broker以group分开,每个group只允许一个master,若干个slave。

只有master才能进行写入操作,slave不允许。

slave从master中同步数据。同步策略取决于master的配置,可以采用同步双写,异步复制两种。

客户端消费可以从master和slave消费。在默认情况下,消费者都从master消费,在master挂后,客户端由于从Name Server中感知到Broker挂机,就会从slave消费。

Broker向所有的NameServer结点建立长连接,注册Topic信息。

RocketMQ优点

1.强调集群无单点,可扩展

2.任意一点高可用,水平可扩展

3.海量消息堆积能力,消息堆积后,写入低延迟。

4.支持上万个队列

5.消息失败重试机制

6.消息可查询

7.开源社区活跃

8.成熟度(经过双十一考验)

RocketMQ环境安装

服务器配置

  1. 192.168.110.187 nameServer1,brokerServer1
  2.  
  3. 192.168.110.188 nameServer2,brokerServer2

添加Host文件

  1. vi /etc/hosts
  2.  
  3. 192.168.110.187 rocketmq-nameserver1
  4.  
  5. 192.168.110.187 rocketmq-master1
  6.  
  7. 192.168.110.188 rocketmq-nameserver2
  8.  
  9. 192.168.110.188 rocketmq-master2
  10.  
  11. service network restart

注意: Error:No suitable device found: no device found for connection "System eth0"

解决办法:

(1)ifconfig -a 查看物理 MAC HWADDR 的值

(2)vim 编辑文件 /etc/sysconfig/network-scripts/ifcfg-eth0中修改ifconfig中查出的MAC HWADDR值;

上传安装包

  1. # 上传alibaba-rocketmq-3.2.6.tar.gz文件至/usr/localtar -zxvf alibaba-rocketmq-3.2.6.tar.gz -C /usr/localmv alibaba-rocketmq alibaba-rocketmq-3.2.6ln -s alibaba-rocketmq-3.2.6 rocketmq

创建存储路径【两台机器】

  1. mkdir /usr/local/rocketmq/store
  2.  
  3. mkdir /usr/local/rocketmq/store/commitlog
  4.  
  5. mkdir /usr/local/rocketmq/store/consumequeue
  6.  
  7. mkdir /usr/local/rocketmq/store/index

RocketMQ配置文件【两台机器】

  1. vim /usr/local/rocketmq/conf/2m-noslave/broker-a.properties
  2.  
  3. vim /usr/local/rocketmq/conf/2m-noslave/broker-b.properties

修改日志配置文件【两台机器】

  1. mkdir -p /usr/local/rocketmq/logs
  2.  
  3. cd /usr/local/rocketmq/conf && sed -i 's#${user.home}#/usr/local/rocketmq#g' *.xml

修改启动NameServer【两台机器】

  1. vim /usr/local/rocketmq/bin/runbroker.sh
  1. JAVA_OPT="${JAVA_OPT} -server -Xms1g -Xmx1g -Xmn512m -
  2. XX:PermSize=128m -XX:MaxPermSize=320m"
  1. vim /usr/local/rocketmq/bin/runserver.sh

JAVA_OPT="${JAVA_OPT} -server -Xms1g -Xmx1g -Xmn512m -

XX:PermSize=128m -XX:MaxPermSize=320m"

启动NameServer【两台机器】

  1. cd /usr/local/rocketmq/binnohup sh mqnamesrv &

启动BrokerServer A

  1. cd /usr/local/rocketmq/bin
  2. nohup sh mqbroker -c /usr/local/rocketmq/conf/2m-noslave/broker-a.properties >/dev/null 2>&1 &netstat -ntlpjpstail -f -n 500 /usr/local/rocketmq/logs/rocketmqlogs/broker.logtail -f -n 500 /usr/local/rocketmq/logs/rocketmqlogs/namesrv.log

启动BrokerServer B

  1. cd /usr/local/rocketmq/bin
  2. nohup sh mqbroker -c /usr/local/rocketmq/conf/2m-noslave/broker-b.properties >/dev/null 2>&1 &netstat -ntlpjpstail -f -n 500 /usr/local/rocketmq/logs/rocketmqlogs/broker.logtail -f -n 500 /usr/local/rocketmq/logs/rocketmqlogs/namesrv.log

RocketMQ Console

将rocketmq-web-console 部署到webapps目录中。

  1. /usr/local/apache-tomcat-7.0.65/webapps/rocketmq-web-console/WEB-INF/classes/

修改config.properties

  1. rocketmq.namesrv.addr=192.168.110.195:9876;192.168.110.199:9876

运行效果

安装jdk环境

  1. vi /etc/profile

export JAVA_HOME=/usr/local/jdk1.7.0_80

export CLASSPATH=.:$JAVA_HOME/lib/dt.jar:$JAVA_HOME/lib/tools.jar

export PATH=$JAVA_HOME/bin:$PATH

  1. source /etc/profile

Java操作RocketMQ

pom文件依赖

  1. <dependencies>
  2. <dependency>
  3. <groupId>com.alibaba.rocketmq</groupId>
  4. <artifactId>rocketmq-client</artifactId>
  5. <version>3.0.10</version>
  6. </dependency>
  7. <dependency>
  8. <groupId>com.alibaba.rocketmq</groupId>
  9. <artifactId>rocketmq-all</artifactId>
  10. <version>3.0.10</version>
  11. <type>pom</type>
  12. </dependency>
  13. <dependency>
  14. <groupId>ch.qos.logback</groupId>
  15. <artifactId>logback-classic</artifactId>
  16. <version>1.1.1</version>
  17. </dependency>
  18. <dependency>
  19. <groupId>ch.qos.logback</groupId>
  20. <artifactId>logback-core</artifactId>
  21. <version>1.1.1</version>
  22. </dependency>
  23. <dependency>
  24. <groupId>junit</groupId>
  25. <artifactId>junit</artifactId>
  26. <version>4.10</version>
  27. <scope>test</scope>
  28. </dependency>
  29. </dependencies>

生产者

  1. package com.hongmoshui;
  2. import com.alibaba.rocketmq.client.exception.MQClientException;
  3. import com.alibaba.rocketmq.client.producer.DefaultMQProducer;
  4. import com.alibaba.rocketmq.client.producer.SendResult;
  5. import com.alibaba.rocketmq.common.message.Message;
  6.  
  7. public class Producer
  8. {
  9. public static void main(String[] args) throws MQClientException
  10. {
  11. DefaultMQProducer producer = new DefaultMQProducer("rmq-group");
  12. producer.setNamesrvAddr("192.168.110.195:9876;192.168.110.199:9876");
  13. producer.setInstanceName("producer");
  14. producer.start();
  15. try
  16. {
  17. for (int i = 0; i < 10; i++)
  18. {
  19. // 每秒发送一次MQ
  20. Thread.sleep(1000);
  21. // topic:主题名称,tag:临时值,body:内容
  22. Message msg = new Message("hongmoshui-topic", "TagA", ("hongmoshui-" + i).getBytes());
  23. SendResult sendResult = producer.send(msg);
  24. System.out.println(sendResult.toString());
  25. }
  26. }
  27. catch (Exception e)
  28. {
  29. e.printStackTrace();
  30. }
  31. producer.shutdown();
  32. }
  33. }

消费者

  1. package com.hongmoshui;
  2.  
  3. import java.util.List;
  4.  
  5. import com.alibaba.rocketmq.client.consumer.DefaultMQPushConsumer;
  6. import com.alibaba.rocketmq.client.consumer.listener.ConsumeConcurrentlyContext;
  7. import com.alibaba.rocketmq.client.consumer.listener.ConsumeConcurrentlyStatus;
  8. import com.alibaba.rocketmq.client.consumer.listener.MessageListenerConcurrently;
  9. import com.alibaba.rocketmq.client.exception.MQClientException;
  10. import com.alibaba.rocketmq.common.message.MessageExt;
  11.  
  12. public class Consumer
  13. {
  14. public static void main(String[] args) throws MQClientException
  15. {
  16. DefaultMQPushConsumer consumer = new DefaultMQPushConsumer("rmq-group");
  17.  
  18. consumer.setNamesrvAddr("192.168.110.195:9876;192.168.110.199:9876");
  19. consumer.setInstanceName("consumer");
  20. consumer.subscribe("hongmoshui-topic", "TagA");
  21.  
  22. consumer.registerMessageListener(new MessageListenerConcurrently()
  23. {
  24. public ConsumeConcurrentlyStatus consumeMessage(List<MessageExt> msgs, ConsumeConcurrentlyContext context)
  25. {
  26. for (MessageExt msg : msgs)
  27. {
  28. System.out.println(msg.getMsgId() + "---" + new String(msg.getBody()));
  29. }
  30. return ConsumeConcurrentlyStatus.CONSUME_SUCCESS;
  31. }
  32. });
  33. consumer.start();
  34. System.out.println("Consumer Started.");
  35. }
  36. }

RocketMQ重试机制

MQ 消费者的消费逻辑失败时,可以通过设置返回状态达到消息重试的结果。

MQ 消息重试只针对集群消费方式生效;广播方式不提供失败重试特性,即消费失败后,失败消息不再重试,继续消费新的消息。

  1. package com.hongmoshui.test2;
  2. import java.util.List;
  3. import com.alibaba.rocketmq.client.consumer.DefaultMQPushConsumer;
  4. import com.alibaba.rocketmq.client.consumer.listener.ConsumeConcurrentlyContext;
  5. import com.alibaba.rocketmq.client.consumer.listener.ConsumeConcurrentlyStatus;
  6. import com.alibaba.rocketmq.client.consumer.listener.MessageListenerConcurrently;
  7. import com.alibaba.rocketmq.client.exception.MQClientException;
  8. import com.alibaba.rocketmq.common.message.MessageExt;
  9.  
  10. public class Consumer
  11. {
  12. public static void main(String[] args) throws MQClientException
  13. {
  14. DefaultMQPushConsumer consumer = new DefaultMQPushConsumer("rmq-group");
  15.  
  16. consumer.setNamesrvAddr("192.168.110.195:9876;192.168.110.199:9876");
  17. consumer.setInstanceName("consumer");
  18. consumer.subscribe("hongmoshui-topic", "TagA");
  19. consumer.registerMessageListener(new MessageListenerConcurrently()
  20. {
  21. public ConsumeConcurrentlyStatus consumeMessage(List<MessageExt> msgs, ConsumeConcurrentlyContext context)
  22. {
  23. for (MessageExt msg : msgs)
  24. {
  25. System.out.println(msg.getMsgId() + "---" + new String(msg.getBody()));
  26. }
  27. try
  28. {
  29. int i = 1 / 0;
  30. }
  31. catch (Exception e)
  32. {
  33. e.printStackTrace();
  34. // 需要重试
  35. return ConsumeConcurrentlyStatus.RECONSUME_LATER;
  36. }
  37. // 不需要重试
  38. return ConsumeConcurrentlyStatus.CONSUME_SUCCESS;
  39. }
  40. });
  41. consumer.start();
  42. System.out.println("Consumer Started.");
  43. }
  44. }

注意:每次重试后,消息ID都不一致,所以不能使用消息ID判断幂等。

RocketMQ如何解决消息幂等

注意:每次重试后,消息ID都不一致,所以不能使用消息ID判断幂等。

解决办法:使用自定义全局ID判断幂等,例如流水ID、订单号

使用msg.setKeys 进行区分

生产者:

  1. package com.hongmoshui.test3;
  2. import com.alibaba.rocketmq.client.exception.MQClientException;
  3. import com.alibaba.rocketmq.client.producer.DefaultMQProducer;
  4. import com.alibaba.rocketmq.client.producer.SendResult;
  5. import com.alibaba.rocketmq.common.message.Message;
  6.  
  7. public class Producer
  8. {
  9. public static void main(String[] args) throws MQClientException
  10. {
  11. DefaultMQProducer producer = new DefaultMQProducer("rmq-group");
  12. producer.setNamesrvAddr("192.168.110.195:9876;192.168.110.199:9876");
  13. producer.setInstanceName("producer");
  14. producer.start();
  15. try
  16. {
  17. for (int i = 0; i < 1; i++)
  18. {
  19. // 每秒发送一次MQ
  20. Thread.sleep(1000);
  21. // topic:主题名称,tag:临时值,body内容
  22. Message msg = new Message("hongmoshui-topic", "TagA", ("hongmoshui-" + i).getBytes());
  23. msg.setKeys(System.currentTimeMillis() + "");
  24. SendResult sendResult = producer.send(msg);
  25. System.out.println(sendResult.toString());
  26. }
  27. }
  28. catch (Exception e)
  29. {
  30. e.printStackTrace();
  31. }
  32. producer.shutdown();
  33. }
  34.  
  35. }

消费者:

  1. package com.hongmoshui.test3;
  2. import java.util.HashMap;
  3. import java.util.List;
  4. import java.util.Map;
  5. import com.alibaba.rocketmq.client.consumer.DefaultMQPushConsumer;
  6. import com.alibaba.rocketmq.client.consumer.listener.ConsumeConcurrentlyContext;
  7. import com.alibaba.rocketmq.client.consumer.listener.ConsumeConcurrentlyStatus;
  8. import com.alibaba.rocketmq.client.consumer.listener.MessageListenerConcurrently;
  9. import com.alibaba.rocketmq.client.exception.MQClientException;
  10. import com.alibaba.rocketmq.common.message.MessageExt;
  11.  
  12. public class Consumer
  13. {
  14. static private Map<String, String> logMap = new HashMap<String, String>();
  15.  
  16. public static void main(String[] args) throws MQClientException
  17. {
  18. DefaultMQPushConsumer consumer = new DefaultMQPushConsumer("rmq-group");
  19.  
  20. consumer.setNamesrvAddr("192.168.110.195:9876;192.168.110.199:9876");
  21. consumer.setInstanceName("consumer");
  22. consumer.subscribe("hongmoshui-topic", "TagA");
  23. consumer.registerMessageListener(new MessageListenerConcurrently()
  24. {
  25. public ConsumeConcurrentlyStatus consumeMessage(List<MessageExt> msgs, ConsumeConcurrentlyContext context)
  26. {
  27. String key = null;
  28. String msgId = null;
  29. try
  30. {
  31. for (MessageExt msg : msgs)
  32. {
  33. key = msg.getKeys();
  34. if (logMap.containsKey(key))
  35. {
  36. // 无需继续重试。
  37. System.out.println("key:" + key + ",无需重试...");
  38. return ConsumeConcurrentlyStatus.CONSUME_SUCCESS;
  39. }
  40. msgId = msg.getMsgId();
  41. System.out.println("key:" + key + ",msgid:" + msgId + "---" + new String(msg.getBody()));
  42. int i = 1 / 0;
  43. }
  44. }
  45. catch (Exception e)
  46. {
  47. e.printStackTrace();
  48. return ConsumeConcurrentlyStatus.RECONSUME_LATER;
  49. }
  50. finally
  51. {
  52. logMap.put(key, msgId);
  53. }
  54. return ConsumeConcurrentlyStatus.CONSUME_SUCCESS;
  55. }
  56. });
  57. consumer.start();
  58. System.out.println("Consumer Started.");
  59. }
  60.  
  61. }

【学习】025 RocketMQ的更多相关文章

  1. rocketmq学习(一) rocketmq介绍与安装

    1.消息队列介绍 消息队列本质上来说是一个符合先进先出原则的单向队列:一方发送消息并存入消息队列尾部(生产者投递消息),一方从消息队列的头部取出消息(消费者消费消息).但对于一个成熟可靠的消息队列来说 ...

  2. rocketmq学习(二) rocketmq集群部署与图形化控制台安装

    1.rocketmq图形化控制台安装 虽然rocketmq为用户提供了使用命令行管理主题.消费组以及broker配置的功能,但对于不够熟练的非运维人员来说,命令行的管理界面还是较难使用的.为此,我们可 ...

  3. RocketMQ事务消息学习及刨坑过程

    一.背景 MQ组件是系统架构里必不可少的一门利器,设计层面可以降低系统耦合度,高并发场景又可以起到削峰填谷的作用,从单体应用到集群部署方案,再到现在的微服务架构,MQ凭借其优秀的性能和高可靠性,得到了 ...

  4. rocketmq 部署启动指南-Docker 版

    最近学习使用 rocketmq,需要搭建 rocketmq 服务端,本文主要记录 rocketmq 搭建过程以及这个过程踩到的一些坑. 准备工作 在搭建之前,我们需要做一些准备工作,这里我们需要使用 ...

  5. RocketMQ支持事务消息机制

    事务消费 我们经常支付宝转账余额宝,这是日常生活的一件普通小事,但是我们思考支付宝扣除转账的钱之后,如果系统挂掉怎么办,这时余额宝账户并没有增加相应的金额,数据就会出现不一致状况了. 上述场景在各个类 ...

  6. Docker 版rocketmq部署

    rocketmq 部署启动指南-Docker 版   最近学习使用 rocketmq,需要搭建 rocketmq 服务端,本文主要记录 rocketmq 搭建过程以及这个过程踩到的一些坑. 准备工作# ...

  7. Spring Cloud Stream消息驱动之RocketMQ入门(一)

    SpringCloudStream目前支持的中间件有RabbitMQ.Kafka,还有我最近在学习的RocketMQ,以下是我学习的笔记 学习Spring cloud Stream 可以先学习一下了解 ...

  8. 面试官:RocketMQ是什么,它有什么特性与使用场景?

    哈喽!大家好,我是小奇,一位热爱分享的程序员 小奇打算以轻松幽默的对话方式来分享一些技术,如果你觉得通过小奇的文章学到了东西,那就给小奇一个赞吧 文章持续更新 一.前言 作为一名Java程序员,Roc ...

  9. MQ系列5:RocketMQ消息的发送模式

    MQ系列1:消息中间件执行原理 MQ系列2:消息中间件的技术选型 MQ系列3:RocketMQ 架构分析 MQ系列4:NameServer 原理解析 在之前的篇章中,我们学习了RocketMQ的原理, ...

随机推荐

  1. python3使用ltp语言云

    text="我爱自然语言处理." text=str(text) #text=urllib.quote(text) text=urllib.parse.quote(text) def ...

  2. InputNumber计数器

    InputNumber 计数器 仅允许输入标准的数字值,可定义范围 要使用它,只需要在el-input-number元素中使用v-model绑定变量即可,变量的初始值即为默认值. <templa ...

  3. Java学习笔记之ArrayList基本用法

    原文地址,转载请注明出处:https://blog.csdn.net/GongchuangSu/article/details/51514389 ArrayList简介 ArrayList是一个其容量 ...

  4. 【HTML5】---【HTML5提供的一些新的标签用法以及和HTML 4的区别】

    HTML 5 是一个新的网络标准,目标在于取代现有的 HTML 4.01, XHTML 1.0 and DOM Level 2 HTML 标准.它希望能够减少浏览器对于需要插件的丰富性网络应用服务(p ...

  5. Web测试方法_01

    一.输入框 1.字符型输入框: (1)字符型输入框:英文全角.英文半角.数字.空或者空格.特殊字符“~!@#¥%……&*?[]{}”特别要注意单引号和&符号.禁止直接输入特殊字符时,使 ...

  6. Redis 入门 3.1 热身

    3.1 热身 1. 获得符合规则的键名列表 KEYS pattern pattern 支持 glob 风格通配符格式 语言 字符组 ? 匹配一个字符 * 匹配任意个(包括0个)字符 [] 匹配括号间的 ...

  7. 【PI系列】SAP IDOC发送状态03,PI没有收到消息的解决办法

    公众号:SAP Technical 本文作者:matinal 原文出处:http://www.cnblogs.com/SAPmatinal/ 原文链接:[PI系列]SAP IDOC发送状态03,PI没 ...

  8. 引用dll出现了黄色感叹号

    今天引用一个dll的时候出现了一个小感叹号,重新生成也无济于事,如下图 原因是,被引用的项目使用的是.NET2.0版本,而当前项目使用的是.NET3.5版本,所以出现了错误 解决办法把当前项目和引用项 ...

  9. numpy伪随机数的生成

    numpy伪随机数的生成 normal函数 可以用normal来得到一个标准正态分布的4×4样本数组 >>> import numpy as np >>> samp ...

  10. springboot2.0结合fastdfs实现文件分布式上传

    1. 引入依赖 在父工程中,我们已经管理了依赖,版本为: <fastDFS.client.version>1.26.7</fastDFS.client.version> 因此, ...