Docker 安装 RocketMQ 并结合 SpringBoot 使用实例

在之前的《浅入浅出消息队列》一文中,我们了解了消息队列的作用、优缺点和使用场景,相信你对消息队列已经有了一个大致的概念,文末给自己埋的坑说日后会写一篇实战教程,正好现在实习结束了,也许久没有写实战教程了,于是这就来填坑了。

前置知识

阅读本文前,建议有一些前置知识,包括且不限于:

  • 常见的 Linux 命令
  • 消息队列的相关知识
  • Docker 的基本使用
  • docker-compose 的基础知识
  • SpringBoot 的基本使用

那废话不多说,我们就开始吧。

本文的所涉及到的代码可在微信公众号「01 二进制」后台回复「rocketmq」获得。

为什么要以 RocketMQ 为例?

本文主要是为了通过实例的方式直观的了解消息队列。那么问题来了,消息队列那么多(ActiveMQ、RabbitMQ、Kafka),为什么要选择 RocketMQ 呢?这里我们不谈原理,只说说体验,仅是个人选择,不喜勿喷。

  1. 背靠阿里,不看测评,纯粹看他经历过多次双十一的检验就已经知道其性能是处于第一批次的。
  2. 作为一个 Java 程序员,如果选择一个纯 Java 编写的软件,后期阅读其源码难度也会小很多。(RabbitMQ 底层是 Erlang,kafka 底层是 Scala)
  3. 在阿里实习的时候一直都是使用 RocketMQ 的内部版本,于我而言,RocketMQ 更熟悉。

初识 RocketMQ

在使用消息队列前,我们要知道消息队列是什么,这一块内容参考之前的文章《浅入浅出消息队列》,这里不再赘述。

本段节来讲解 RocketMQ 所涉及到的相关概念,我们先来简单看下官方给出的 RocketMQ 架构图

image

从上图我们可以很直观的看出,一个完整的 RocketMQ 架构包含四个部分:NameServer、Broker、Producer 和 Consumer

  • NameServer:主要用作注册中心,用于管理 Topic 信息和路由信息的管理
  • Broker:负责存储、消息 tag 过滤和转发。需将自身信息上报给注册中心 NameServer
  • Producer:生产者
  • Consumer:消费者

从寄信的角度理解

上面的解释可能难以理解,我们从寄信这一实例来看以下四个部分所承担的责任。

  • Producer 和 Consumer 不必多说,消息的生产者和消费者,生产者负责投递消息,消费者负责接收消息,是我们要编写的应用程序。可以理解为寄信人和收信人。
  • Broker 负责消息存储,以 Topic(主题)为维度,以队列的形式存储消息。可以理解为信箱,专门存储信件,收信人(Consumer)可以从这里获取信件。
  • NameServer 负责对源数据进行管理,包括了对 Topic 和 Broker 的管理。可以理解为邮局,负责管理邮件的分发,维护信箱(Broker)的状态。

由上各部分角色的功能可知,我们需要先安装启动 NameServer,再启动 Broker 即可搭建完 RocketMQ

安装 RocketMQ

如果你的电脑上已经配置好了 rocketmq 的相关环境,可以跳过本章节。

从上面的介绍我们可以得知,在生产和消费消息之前,我们需要安装好Broker 和 NameServer。

准备工作

为了部署方便,我推荐使用 docker 搭建服务。此外,由于 rocketmq 需要分别部署 broker 与 nameserver ,考虑到分开部署比较麻烦,这里我将会使用 docker-compose。因此,你需要在你的宿主机中安装好 docker 和 docker-compose。

此外,我们还需要搭建一个 web 可视化控制台,用于监控 mq 服务状态,以及消息消费情况,这里使用 rocketmq-console,同样该程序也将使用 docker 安装。

如果对 docker 不熟悉的话,可以先阅读菜鸟教程的 docker 教程学习 👉Docker 教程

安装

安装 Docker

Linux:

执行以下命令

curl -fsSL https://get.docker.com | bash -s docker --mirror Aliyun

Mac:

执行以下命令

brew cask install docker

Win:

下载对应的安装文件,然后双击运行安装。下载地址在:https://hub.docker.com/editions/community/docker-ce-desktop-windows

考虑到下载该文件需要科学上网,你可以在微信公众号「01 二进制」后台回复「docker」获取 docker 安装包的下载链接。

如果你的 win10 系统可以使用 winget,那就执行以下命令。(win 终于也有自己的包管理工具了 🙏)

winget install Docker.DockerDesktop

国内从 DockerHub 拉取镜像有时会遇到困难,此时可以配置镜像加速器。配置教程可参考 👉Docker 镜像加速

安装 RocketMQ 镜像

rocketmq 的 docker 镜像我们可以自己制作,官方文档中有详细介绍 👉apache/rocketmq-docker

为了方便起见,这里我们直接使用别人已经制作好的镜像,镜像地址 👉 foxiswho/rocketmq

新建一个目录用于存放相关脚本,然后在终端执行下面的命令 👇

git clone https://github.com/foxiswho/docker-rocketmq.git
cd docker-rocketmq
cd rmq
chmod +x  start.sh
./start.sh

在经过一段时间的等待后,我们通过浏览器访问localhost:8180查看到以下页面则说明安装成功。

image

安装脚本解析

通过脚本的方式一键安装确实很方便,但如果只是安装完成就万事大吉了自然是不行的,本着授人以渔的态度,我们来看看安装脚本里都有些啥:

start.sh

image

4-7 行在创建目录,10-13 行在给刚才创建的目录设置权限,至于原因我们之后再说。

我们看到 16 行使用 docker-compose 命令启动了容器,并设置为了后台自动启动,因此我们来看一下这个 docker-compose.yml 文件。

docker-compose.yml

version: "3.5"

services:
  rmqnamesrv:
    image: foxiswho/rocketmq:4.7.0
    container_name: rmqnamesrv
    ports:
      - 9876:9876
    volumes:
      - ./rmqs/logs:/opt/logs
      - ./rmqs/store:/opt/store
    environment:
      JAVA_OPT_EXT: "-Duser.home=/opt -Xms512M -Xmx512M -Xmn128m"
    command: ["sh", "mqnamesrv"]
    networks:
      rmq:
        aliases:
          - rmqnamesrv
  rmqbroker:
    image: foxiswho/rocketmq:4.7.0
    container_name: rmqbroker
    ports:
      - 10909:10909
      - 10911:10911
    volumes:
      - ./rmq/logs:/opt/logs
      - ./rmq/store:/opt/store
      - ./rmq/brokerconf/broker.conf:/etc/rocketmq/broker.conf
    environment:
      JAVA_OPT_EXT: "-Duser.home=/opt -Xms512M -Xmx512M -Xmn128m"
    command:
      [
        "sh",
        "mqbroker",
        "-c",
        "/etc/rocketmq/broker.conf",
        "-n",
        "rmqnamesrv:9876",
        "autoCreateTopicEnable=true",
      ]
    depends_on:
      - rmqnamesrv
    networks:
      rmq:
        aliases:
          - rmqbroker

  rmqconsole:
    image: styletang/rocketmq-console-ng
    container_name: rmqconsole
    ports:
      - 8180:8080
    environment:
      JAVA_OPTS: "-Drocketmq.namesrv.addr=rmqnamesrv:9876 -Dcom.rocketmq.sendMessageWithVIPChannel=false"
    depends_on:
      - rmqnamesrv
    networks:
      rmq:
        aliases:
          - rmqconsole

networks:
  rmq:
    name: rmq
    driver: bridge

我们创建了三个服务,这三个服务的名字分别是 rmqnamesrv、rmqbroker 和 rmqconsole,分别对应我们之前所说的 nameserver、broker 和可视化控制台。并且对不同的服务做了不同的端口映射,同时将本地指定的文件目录挂载到 docker 容器中,并以网桥(bridge)的形式进行网络连接。

rmqnamesrv为例,其基础镜像为foxiswho/rocketmq:4.7.0,创建的容器名为rmqnamesrv,并将其内部的 9876 端口映射到宿主机的 9876 端口,并将本地的./rmqs/logs文件挂载到 docker 容器的/opt/logs目录中。

rmqnamesrv:
  image: foxiswho/rocketmq:4.7.0
  container_name: rmqnamesrv
  ports:
    - 9876:9876
  volumes:
    - ./rmqs/logs:/opt/logs
    - ./rmqs/store:/opt/store

如果对于 docker-compose 不熟悉的读者,可以先参考相关的教程学习一下 👉Docker Compose

SpringBoot 整合 RocketMQ 小实例

在完成了相对复杂的安装、配置后,我们终于可以实现一个小的 demo 来打通整个流程了。

创建消息主题和订阅组

使用 RocketMQ 进行发消息时,必须要指定 topic,对于 topic 的设置有一个开关autoCreateTopicEnable,一般在开发测试环境中会使用默认设置autoCreateTopicEnable = true,但是这样就会导致 topic 的设置不容易规范管理,没有统一的审核等等,所以在正式环境中会在 Broker 启动时设置参数autoCreateTopicEnable = false。这样当需要增加 topic 时就需要在 web 管理界面上添加即可。

在 web 界面添加 topic 的方式如下:

image

同理,在接受消息时,我们同样需要对消息订阅组进行配置,对于消息的订阅设置有一个开关autoCreateSubscriptionGroup,通常情况下,在生产环境下,我们需要设置为autoCreateSubscriptionGroup=false,这就要求了管理者必须去 web 管理界面上创建订阅组才可以收到消息。

在 web 界面添加订阅组的方式类似,如下图所示:

image

如果只是测试环境,我们可以在配置文件中将这两个开关打开,配置文件在 rmq/rmq/brokerconf 目录下

编写代码

apache 官方已经提供了 rocketmq 对应的 springboot starter,这极大的简化了我们所需要做的配置工作,因此我们要做的就是先新建一个 springboot 项目,然后按照下面的方式着手实现。

导入依赖

首先先在 pom.xml 中导入 apache 官方提供的 starter

<!-- https://mvnrepository.com/artifact/org.apache.rocketmq/rocketmq-spring-boot-starter -->
<dependency>
    <groupId>org.apache.rocketmq</groupId>
    <artifactId>rocketmq-spring-boot-starter</artifactId>
    <version>2.1.0</version>
</dependency>

配置 application.yml

依赖导入后,我们需要在 application.yml 配置一个 name-server 地址,具体值看你的机器。

rocketmq:
  name-server: localhost:9876
  producer:
    group: myGroup

创建一个生产者类

生产者发送消息:

@RestController
public class RocketController {
    @Autowired
    private RocketMQTemplate rocketMQTemplate;
    // 发送给Broker,默认会自动创建topic,topic和tag用冒号分隔
    @GetMapping("/rocket/send")
    public String rocketSend() {
        LocalDateTime currentTime = LocalDateTime.now();
        rocketMQTemplate.convertAndSend("rocket-topic-2", currentTime.toString());
        return currentTime.toString();
    }
    // 延时消息,RocketMQ支持这几个级别的延时消息,不能自定义时长
    // 1s 5s 10s 30s 1m 2m 3m 4m 5m 6m 7m 8m 9m 10m 20m 30m 1h 2h
    @GetMapping("/rocket/delayMsg/send")
    public String rocketDelayMsgSend() {
        LocalDateTime currentTime = LocalDateTime.now();
        rocketMQTemplate.syncSend("rocket-topic-2:tag-2", MessageBuilder.withPayload(currentTime.toString()).build(), 2000, 3);
        return currentTime.toString();
    }
}

创建一个消费者

消费者监听消息:

@Component
@Slf4j
public class RokcetServiceListener {
    @Service
    @RocketMQMessageListener(consumerGroup = "consumer-group-1", topic = "rocket-topic-2")
    public class Consumer1 implements RocketMQListener<String> {
        @Override
        public void onMessage(String s) {
            log.info("consumer1 rocket收到消息:{}", s);
        }
    }
    // RocketMQ支持两种消费方式,集器消费和广播消费
    @Service
    @RocketMQMessageListener(consumerGroup = "consumer-group-2", topic = "rocket-topic-2",
            selectorExpression = "tag2", messageModel = MessageModel.BROADCASTING)
    public class Consumer2 implements RocketMQListener<String> {
        @Override
        public void onMessage(String s) {
            log.info("consumer2 rocket收到消息:{}", s);
        }
    }
}

测试

我们在浏览器中访问localhost:8080/rocket/send,即可看到返回的时间戳。

image

同时在控制台也可以看到消费者已经获取到这条信息了

image

同样的,我们也可以在可视化控制台查看到相应的消息

image

我们同样可以在可视化控制台查看消费者和生产者对于消息的生产与消费的情况,这些就留给读者自己探索了。至此,一个完整的利用 Docker 安装 RocketMQ 并结合 SpringBoot 使用的实例就结束了。

问题

问题 1:No route info of this topic: xxxxxx

通过翻译我们可以知道,这个错误产生的原因是因为消息队列中并未产生相对应的topic,所以我们要做的应该是去控制台新建一个 topic

image

问题 2:连接异常

如果出现类似下述这种连接异常的错误

com.alibaba.rocketmq.remoting.exception.RemotingConnectException: connect to <172.0.0.120:10909> failed

可能的原因是你并没有将项目放至 docker 容器中,因此你的项目代码不能直接与 rocketmq 容器访问,因此我们需要将broker.conf中的 #brokerIP1=xxxxx 前面#号去掉,并且把后面的IP地址改成你的rocketmq容器宿主机IP地址,配置文件在 rmq/rmq/brokerconf 目录下。

最后

为了填坑,我选择了 rocketmq 作为实例讲解的对象,并在第一节阐述了我为什么要使用 RocketMQ 的原因,之后解释了 RocketMQ 中几个重要的概念,然后利用 docker 快速的部署安装了一个 rocketmq 的单机实例,并分析了安装脚本。最后我们通过 springboot 这一目前主流的 web 框架实现了一个生产者与消费者的实例,并说明了可能会遇到的问题及解决方案。

以上就是本文的全部内容了,如果你觉得对你有所帮助,不放关注点赞支持一波,你们的支持是我更新的最大动力。

作者:雇个城管打天下
链接:https://www.jianshu.com/p/9e5437eb6939

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,444评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,421评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,363评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,460评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,502评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,511评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,280评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,736评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,014评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,190评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,848评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,531评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,159评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,411评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,067评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,078评论 2 352

推荐阅读更多精彩内容