辛星RocketMQ教程第一篇:第一个范例

[前言:本教程基于RocketMQ4.2,项目基于maven构建,使用操作系统为*nix]

今天我们要介绍的就是RocketMQ,它是阿里巴巴在2012年开源的分布式消息中间件,目前已经捐赠给Apache基金会,在2016年11月成为Apache孵化项目。

在最开始,来介绍一下什么是消息队列,比如一个用户注册后,我们可能需要更新很多统计信息,比如用户总数、当天注册用户数、当月注册用户数,有些网站还会给这个用户发邮件,会给用户注册im账号,这些操作可能比较耗时,那怎么办呢?

我们可以通过消息队列来进行解耦,在用户注册后,我们把用户的信息发送到消息队列中,然后其他功能只需要不断的从消息队列中获取数据,然后进行相关的操作即可。

这里可以看到,消息队列至少有两个好处:
1.解耦了各个模块,让各个模块可以独立开发和部署。
2.减少主流程的步骤,非核心流程操作不会阻塞主流程。

那么都有哪些消息队列可用呢?最简单的消息中间件可能要数Redis,但是它不是一个传统意义上的消息队列,它缺少可靠性保证,它更注重的是性能,但是传统意义上的消息队列不仅要注重性能,还要关注其他几个点。

消息队列需要关注的点有:
第一,消息的可靠性。如果纯内存的消息队列,那么就可能会因为程序突然中止而导致数据丢失,所以一般消息队列都会记录到磁盘来保证其可靠性。
第二,消息的顺序。比如我们发送两个消息,比如消息a是修改用户名为aa,消息b是修改用户名为bb,正确的顺序是先发送消息a,然后发送消息b,如果我们的发送顺序是正确的,但是我们的消息队列因为网络等原因让消息b先到达,那么就会产生未知的bug。

首先来科普几个基础概念:
第一个概念即topic,也就是"主题",它是用来区分消息的类型,比如我们在用户注册时发的消息和用户退出时发的消息,应该是两种类型的消息,这两种消息是不相关的,我们如何区分它们呢?我们可以通过"主题"来区分不同的消息。
第二个概念即producer,也就是"生产者",它用来生产消息。比如我们要在用户注册时发一个消息,那么我们就可以写一个SignupProducer来作为消息的生产者。
第三个概念即consumer,也就是"消费者",它用来消费消息。比如我们要在用户注册的时候修改统计数据,我们就可以把这个代码写到SignupConsumer来作为消息的消费者。
第四个概念即Broker,它相当于一个中转站,它从生产者获取消息,然后把消息分发到消费者,需要说明的是,broker需要有一些确认机制,它需要确保我们的消费者正确的收到消息。

对于RocketMQ来说,还有一个概念,即NameServer,它是命名服务器,即RocketMQ的寻址服务,它用于把Broker的路由信息做聚合,客户端通过命名服务器来获取指定topic的路由信息,从而决定对哪些broker做连接。

理论说了那么多,我们首先来下载一下RocketMQ吧,我们可以从http://rocketmq.apache.org/ 这个官网下载,然后我们解压缩之后,我们来看一下bin目录的内容吧:

xinguimeng ~/work/rocketmq-all-4.2.0-bin-release/bin $ ls
README.md       mqbroker        mqfiltersrv     mqshutdown.cmd      runserver.cmd
cachedog.sh     mqbroker.cmd        mqfiltersrv.cmd     os.sh           runserver.sh
cleancache.sh       mqbroker.numanode0  mqfiltersrv.xml     play.cmd        setcache.sh
cleancache.v1.sh    mqbroker.numanode1  mqnamesrv       play.sh         startfsrv.sh
mqadmin         mqbroker.numanode2  mqnamesrv.cmd       runbroker       tools.cmd
mqadmin.cmd     mqbroker.numanode3  mqnamesrv.xml       runbroker.cmd       tools.sh
mqadmin.xml     mqbroker.xml        mqshutdown      runbroker.sh
xinguimeng ~/work/rocketmq-all-4.2.0-bin-release/bin $

我们首先启动命名服务器,我们操作如下:

xinguimeng ~/work/rocketmq-all-4.2.0-bin-release/bin $ sh mqnamesrv
Java HotSpot(TM) 64-Bit Server VM warning: Using the DefNew young collector with the CMS collector is deprecated and will likely be removed in a future release
Java HotSpot(TM) 64-Bit Server VM warning: UseCMSCompactAtFullCollection is deprecated and will likely be removed in a future release.
Java HotSpot(TM) 64-Bit Server VM warning: Cannot open file /dev/shm/rmq_srv_gc.log due to No such file or directory

The Name Server boot success. serializeType=JSON

这里会有报警,我们可以先忽略,然后我们启动broker,如下所示:

xinguimeng ~/work/rocketmq-all-4.2.0-bin-release/bin $ sh mqbroker -n "127.0.0.1:9876"
Java HotSpot(TM) 64-Bit Server VM warning: Cannot open file /dev/shm/mq_gc_pid61639.log due to No such file or directory

在我们启动完毕之后,我们可以新建一个maven项目,我们再pom.xm中写入如下内容:

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>com.mengzhidu</groupId>
    <artifactId>rocketmq-demo</artifactId>
    <version>1.0-SNAPSHOT</version>

    <dependencies>
        <dependency>
            <groupId>com.alibaba.rocketmq</groupId>
            <artifactId>rocketmq-client</artifactId>
            <version>3.2.6</version>
        </dependency>
    </dependencies>

</project>

然后我们新建一个生产者范例,我们写入如下代码:

package com.mengzhidiu.rocketmq.demo;

import com.alibaba.rocketmq.client.producer.DefaultMQProducer;
import com.alibaba.rocketmq.client.producer.SendResult;
import com.alibaba.rocketmq.common.message.Message;

/**
 * 生产者范例
 */
public class ProducerDemo {

    public static void main(String[] args) throws Exception {
        DefaultMQProducer producer = new DefaultMQProducer("mengzhidu-user");
        producer.setNamesrvAddr("127.0.0.1:9876");
        producer.start();

        for (int i = 0; i < 5; i ++) {
            Message message = new Message("user", "push", String.valueOf(i), new String("辛星-" + i).getBytes());
            SendResult result = producer.send(message);
            System.out.println("消息id为:  " + result.getMsgId() + "  发送状态为:" + result.getSendStatus());
        }

    }
}

在生产者范例中,我们制定了消费组为"mengzhidu-user",并且我们制定了命名服务器的地址,然后我们就开始了生产者,然后我们在一个for循环中发送了五次消息,然后我们在生产者中获取了消息的状态。

然后我们新建一个消费者范例,我们写入如下代码:

package com.mengzhidiu.rocketmq.demo;

import com.alibaba.rocketmq.client.consumer.DefaultMQPushConsumer;
import com.alibaba.rocketmq.client.consumer.listener.ConsumeConcurrentlyContext;
import com.alibaba.rocketmq.client.consumer.listener.ConsumeConcurrentlyStatus;
import com.alibaba.rocketmq.client.consumer.listener.MessageListenerConcurrently;
import com.alibaba.rocketmq.client.exception.MQClientException;
import com.alibaba.rocketmq.common.consumer.ConsumeFromWhere;
import com.alibaba.rocketmq.common.message.Message;
import com.alibaba.rocketmq.common.message.MessageExt;

import java.util.List;

/**
 * 消费者范例
 */
public class ConsumerDemo {
    public static void main(String[] args) throws MQClientException {
        DefaultMQPushConsumer consumer = new DefaultMQPushConsumer("mengzhidu-user");
        consumer.setNamesrvAddr("127.0.0.1:9876");
        consumer.subscribe("user", "push");
        consumer.setConsumeFromWhere(ConsumeFromWhere.CONSUME_FROM_FIRST_OFFSET);
        consumer.registerMessageListener(new MessageListenerConcurrently() {
            public ConsumeConcurrentlyStatus consumeMessage(List<MessageExt> list,
                                                            ConsumeConcurrentlyContext consumeConcurrentlyContext) {
                Message message = list.get(0);
                System.out.println("消费者收到消息的内容:" + new String(message.getBody()));
                return ConsumeConcurrentlyStatus.CONSUME_SUCCESS;
            }
        });

        consumer.start();
    }
}

我们的消费者也制定了主题为"mengzhidu-user",而且它也指定了命名服务器的地址,然后设置了在收到消息后的处理方式,然后就启动了消费者。

然后我们首先启动消费者,然后我们启动生产者,在一小段时间后,我们会看到消费者端打印如下:

消费者收到消息的内容:辛星-1
消费者收到消息的内容:辛星-2
消费者收到消息的内容:辛星-0
消费者收到消息的内容:辛星-3
消费者收到消息的内容:辛星-4

而生产者端打印如下:

消息id为:  C0A81F5200002A9F0000000000000CB2  发送状态为:SEND_OK
消息id为:  C0A81F5200002A9F0000000000000D34  发送状态为:SEND_OK
消息id为:  C0A81F5200002A9F0000000000000DB6  发送状态为:SEND_OK
消息id为:  C0A81F5200002A9F0000000000000E38  发送状态为:SEND_OK
消息id为:  C0A81F5200002A9F0000000000000EBA  发送状态为:SEND_OK

至此,我们的第一个基于RocketMQ的例子就完成了,不过我们的操作略显粗糙,在之后,我们会做得更加细腻一些。

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

推荐阅读更多精彩内容