继续浏览精彩内容
慕课网APP
程序员的梦工厂
打开
继续
感谢您的支持,我会继续努力的
赞赏金额会直接到老师账户
将二维码发送给自己后长按识别
微信支付
支付宝支付

RocketMQ初识:新手入门指南

斯蒂芬大帝
关注TA
已关注
手记 251
粉丝 7
获赞 21
概述

RocketMQ是一款由阿里巴巴开源的高效分布式消息中间件,具备高性能和高可靠性。本文将详细介绍RocketMQ的核心特性和应用场景,帮助读者快速掌握RocketMQ的基本知识。

RocketMQ简介

RocketMQ是什么

RocketMQ是由阿里巴巴开源的一款分布式消息中间件。它基于Java语言开发,具备高性能、高可靠、高可用的特性,适用于大规模分布式系统的高效、可靠的消息传递需求。RocketMQ的设计目标是满足阿里巴巴集团内部复杂业务场景下的需求,并且已经在阿里巴巴集团内部广泛使用多年。

RocketMQ的核心特性

  1. 高性能: RocketMQ在多个场景下都体现出其高性能的特点,例如在单机每秒能处理数百万的消息量。
  2. 可靠性: RocketMQ保证消息的可靠传递,能够支持禁用消息、事务消息等多种消息类型。
  3. 高可用性: RocketMQ集群的高可用性通过数据多副本复制、定时心跳检查、主从切换等机制来实现。
  4. 扩展性: RocketMQ支持水平扩展,可以轻松地增加或减少服务器节点来满足不同的业务需求。
  5. 灵活性: RocketMQ支持多种消息模型,例如发布/订阅模型、队列模型、顺序消息模型等。

RocketMQ的应用场景

  1. 异步通信: 在分布式系统中,RocketMQ可以作为异步通信的桥梁,实现服务之间的解耦。
  2. 削峰填谷: RocketMQ可以缓解服务之间的高峰压力,通过存储消息来实现削峰填谷。
  3. 数据同步: 在分布式系统中,RocketMQ可以用于数据同步,确保数据的一致性。
  4. 事件驱动: RocketMQ可以作为事件驱动架构的中心,处理复杂的业务逻辑。
  5. 日志收集与分析: RocketMQ可以用于收集和分析日志,实现大规模的日志系统。
环境搭建

下载RocketMQ

  1. 访问RocketMQ的GitHub仓库页面。
  2. 点击“Clone or download”按钮,选择“Download ZIP”下载RocketMQ的压缩包。
  3. 将下载好的压缩包解压到指定目录。

安装RocketMQ

  1. 解压RocketMQ压缩包,运行bin/mq.sh脚本。
  2. 使用./mq.sh命令执行初始化脚本,进行必要的环境配置。
  3. 确保Java环境已正确安装,RocketMQ要求运行环境至少是JDK1.8。

启动RocketMQ

  1. 打开命令行窗口。
  2. 进入RocketMQ的解压目录。
  3. 使用./mqbroker.sh start启动Broker进程。
  4. 使用./mqnamesrv.sh start启动NameServer进程。
  5. 使用./mqadmin tool -n localhost:9876命令查看RocketMQ服务的状态,确保NameServer和Broker服务均已成功启动。
快速入门

发送消息

发送消息是RocketMQ的基本操作之一。首先创建一个Producer实例,然后通过指定的Topic和标签进行消息的发送。

import org.apache.rocketmq.client.producer.DefaultMQProducer;
import org.apache.rocketmq.client.producer.SendResult;
import org.apache.rocketmq.common.message.Message;

public class SendMessage {
    public static void main(String[] args) throws Exception {
        // 创建一个Producer实例
        DefaultMQProducer producer = new DefaultMQProducer("ProducerGroupName");
        // 设置NameServer地址
        producer.setNamesrvAddr("localhost:9876");
        // 启动Producer实例
        producer.start();
        // 构造消息,参数分别是:主题Topic、标签Tag、消息体、消息体的编码
        Message msg = new Message("TopicTest",
                "TagA",
                ("Hello RocketMQ").getBytes(RemotingHelper.DEFAULT_CHARSET));
        // 发送消息到一个单向Topic,无需等待消息发送结果
        SendResult sendResult = producer.send(msg);
        // 输出发送结果
        System.out.println(sendResult);
        // 关闭Producer实例
        producer.shutdown();
    }
}

接收消息

接收消息是RocketMQ的另一基本操作。接收消息是通过创建一个Consumer实例来实现的,通过订阅指定的Topic和标签来接收消息。

import org.apache.rocketmq.client.consumer.DefaultMQPushConsumer;
import org.apache.rocketmq.client.consumer.listener.ConsumeOrderlyContext;
import org.apache.rocketmq.client.consumer.listener.ConsumeOrderlyStatus;
import org.apache.rocketmq.client.consumer.listener.MessageListenerOrderly;
import org.apache.rocketmq.common.consumer.ConsumeFromWhere;
import org.apache.rocketmq.common.message.MessageExt;

public class ReceiveMessage {
    public static void main(String[] args) throws Exception {
        // 创建一个Consumer实例
        DefaultMQPushConsumer consumer = new DefaultMQPushConsumer("ConsumerGroupName");
        // 设置NameServer地址
        consumer.setNamesrvAddr("localhost:9876");
        // 订阅一个或多个Topic,回调方法来处理从broker拉取的消息
        consumer.subscribe("TopicTest", "TagA");
        // 设置消息回溯模式,从最后一次消费的位置开始消费
        consumer.setConsumeFromWhere(ConsumeFromWhere.CONSUME_FROM_LAST_OFFSET);
        // 注册消息监听器
        consumer.registerMessageListener((msgs, context) -> {
            for (MessageExt msg : msgs) {
                // 输出接收到的消息
                System.out.printf("Receive New Messages: %s %n", msg);
            }
            return ConsumeOrderlyStatus.SUCCESS;
        });
        // 启动Consumer实例
        consumer.start();
        // 保持程序运行,以便持续接收消息
        while (true) {
            Thread.sleep(10000);
        }
    }
}

消息模型简述

RocketMQ支持多种消息模型,主要包括:

  1. 发布/订阅模型(Publish/Subscribe): 发布者(Producer)将消息发布到Topic,而订阅者(Consumer)则订阅该Topic上的消息。

    // 发布/订阅模型示例
    public class PublishSubscribeExample {
       public static void main(String[] args) throws Exception {
           // 创建Producer实例
           DefaultMQProducer producer = new DefaultMQProducer("ProducerGroup");
           producer.setNamesrvAddr("localhost:9876");
           producer.start();
           // 创建Consumer实例
           DefaultMQPushConsumer consumer = new DefaultMQPushConsumer("ConsumerGroup");
           consumer.setNamesrvAddr("localhost:9876");
           consumer.subscribe("TopicTest", "TagA");
           consumer.registerMessageListener((msgs, context) -> {
               for (MessageExt msg : msgs) {
                   System.out.println("Message received: " + msg);
               }
               return ConsumeOrderlyStatus.SUCCESS;
           });
           consumer.start();
           // 发送消息
           Message msg = new Message("TopicTest", "TagA", "Hello, World".getBytes(RemotingHelper.DEFAULT_CHARSET));
           SendResult result = producer.send(msg);
           System.out.println("Message sent: " + result);
           Thread.sleep(10000);
           producer.shutdown();
       }
    }
  2. 队列模型(Queue): 每个Topic可以被拆分成多个队列,消息在这些队列之间均匀分布,确保消息的公平调度。

    // 队列模型示例
    public class QueueModelExample {
       public static void main(String[] args) throws Exception {
           // 创建Producer实例
           DefaultMQProducer producer = new DefaultMQProducer("ProducerGroup");
           producer.setNamesrvAddr("localhost:9876");
           producer.start();
           // 创建Consumer实例
           DefaultMQPushConsumer consumer = new DefaultMQPushConsumer("ConsumerGroup");
           consumer.setNamesrvAddr("localhost:9876");
           consumer.subscribe("TopicTest", "TagA");
           consumer.registerMessageListener((msgs, context) -> {
               for (MessageExt msg : msgs) {
                   System.out.println("Message received: " + msg);
               }
               return ConsumeOrderlyStatus.SUCCESS;
           });
           consumer.start();
           // 发送消息
           Message msg = new Message("TopicTest", "TagA", "Hello, World".getBytes(RemotingHelper.DEFAULT_CHARSET));
           SendResult result = producer.send(msg);
           System.out.println("Message sent: " + result);
           Thread.sleep(10000);
           producer.shutdown();
       }
    }
  3. 顺序消息模型(Orderly): 保证消息的有序性,即消息在某个消费者组内按照发送顺序进行消费。

    // 顺序消息模型示例
    public class OrderlyModelExample {
       public static void main(String[] args) throws Exception {
           // 创建Producer实例
           DefaultMQProducer producer = new DefaultMQProducer("ProducerGroup");
           producer.setNamesrvAddr("localhost:9876");
           producer.start();
           // 创建Consumer实例
           DefaultMQPushConsumer consumer = new DefaultMQPushConsumer("ConsumerGroup");
           consumer.setNamesrvAddr("localhost:9876");
           consumer.subscribe("TopicTest", "TagA");
           consumer.registerMessageListener((msgs, context) -> {
               for (MessageExt msg : msgs) {
                   System.out.println("Message received: " + msg);
               }
               return ConsumeOrderlyStatus.SUCCESS;
           });
           consumer.start();
           // 发送消息
           Message msg = new Message("TopicTest", "TagA", "Hello, World".getBytes(RemotingHelper.DEFAULT_CHARSET));
           SendResult result = producer.send(msg);
           System.out.println("Message sent: " + result);
           Thread.sleep(10000);
           producer.shutdown();
       }
    }
常见问题解答

常见错误及解决方案

  1. 消息未被消费:

    • 问题描述: 发送的消息没有被消费者接收到。
    • 解决方案: 检查Consumer的订阅配置是否正确,确保消息的Topic和Tag与发送时一致。
    • 示例代码:
      consumer.subscribe("TopicTest", "TagA");
  2. 消息堆积:

    • 问题描述: 消息队列中积压了大量的未处理消息。
    • 解决方案: 增加Consumer的数量以提高消息处理能力,或者优化消费端的处理逻辑,例如使用批处理减少单条消息处理时间。
    • 示例代码:
      // 消息堆积示例
      public class HandleMessagePiling {
       public static void main(String[] args) throws Exception {
           // 创建Consumer实例
           DefaultMQPushConsumer consumer = new DefaultMQPushConsumer("ConsumerGroup");
           consumer.setNamesrvAddr("localhost:9876");
           consumer.subscribe("TopicTest", "TagA");
           consumer.registerMessageListener((msgs, context) -> {
               for (MessageExt msg : msgs) {
                   // 模拟耗时处理
                   Thread.sleep(5000);
                   System.out.println("Message received: " + msg);
               }
               return ConsumeOrderlyStatus.SUCCESS;
           });
           consumer.start();
           // 模拟消息堆积
           for (int i = 0; i < 100; i++) {
               // 发送消息
               Message msg = new Message("TopicTest", "TagA", ("Message " + i).getBytes(RemotingHelper.DEFAULT_CHARSET));
               SendResult result = producer.send(msg);
               System.out.println("Message sent: " + result);
           }
           Thread.sleep(10000);
           consumer.shutdown();
       }
      }
  3. 发送失败:
    • 问题描述: 发送消息时出现失败。
    • 解决方案: 检查网络连接,确保NameServer和Broker服务运行正常。
    • 示例代码:
      SendResult sendResult = producer.send(msg);
      if (sendResult.getSendStatus() != SendStatus.SEND_OK) {
       System.out.println("发送失败");
      }

常用命令及配置参数

  1. 启动NameServer:
    • 命令: ./mqnamesrv.sh start
  2. 启动Broker:
    • 命令: ./mqbroker.sh start
  3. 查询Broker状态:
    • 命令: ./mqadmin brokerList -n localhost:9876
  4. 配置Broker:
    • 配置文件broker.conf中可以设置如下参数:
      # broker.conf
      brokerName=broker0
      brokerId=0
      namesrvAddr=localhost:9876
      logFile=/path/to/logfile.log
  5. 配置Consumer:
    • consumer.properties文件中可以设置如下参数:
      # consumer.properties
      consumerGroup=logConsumer
      msgModel=Cluster
      unitMode=true
      nameServerAddr=localhost:9876
实践案例

小项目实战

假设我们需要构建一个简单的日志系统,该系统能够接收各个服务上报的日志,然后将日志转发到不同的处理中心进行处理。

发送日志

服务端会将日志消息发送到RocketMQ。

import org.apache.rocketmq.client.producer.DefaultMQProducer;
import org.apache.rocketmq.client.producer.SendResult;
import org.apache.rocketmq.common.message.Message;

public class LogProducer {
    public static void main(String[] args) throws Exception {
        DefaultMQProducer producer = new DefaultMQProducer("logProducer");
        producer.setNamesrvAddr("localhost:9876");
        producer.start();
        for (int i = 0; i < 10; i++) {
            Message msg = new Message("logTopic",
                    "TagA",
                    ("log-" + i).getBytes(RemotingHelper.DEFAULT_CHARSET));
            SendResult sendResult = producer.send(msg);
            System.out.println("Message sent: " + sendResult);
        }
        producer.shutdown();
    }
}

接收日志

日志处理中心接收RocketMQ中发布的日志消息。

import org.apache.rocketmq.client.consumer.DefaultMQPushConsumer;
import org.apache.rocketmq.client.consumer.listener.MessageListenerOrderly;
import org.apache.rocketmq.common.consumer.ConsumeFromWhere;
import org.apache.rocketmq.common.message.MessageExt;

public class LogConsumer {
    public static void main(String[] args) throws Exception {
        DefaultMQPushConsumer consumer = new DefaultMQPushConsumer("logConsumer");
        consumer.setNamesrvAddr("localhost:9876");
        consumer.subscribe("logTopic", "TagA");
        consumer.setConsumeFromWhere(ConsumeFromWhere.CONSUME_FROM_LAST_OFFSET);
        consumer.registerMessageListener((msgs, context) -> {
            for (MessageExt msg : msgs) {
                System.out.printf("Received log: %s %n", msg);
            }
            return ConsumeOrderlyStatus.SUCCESS;
        });
        consumer.start();
        while (true) {
            Thread.sleep(10000);
        }
    }
}

分布式场景下的RocketMQ使用

在分布式环境下,RocketMQ可以用于实现服务之间的异步通信。例如,在订单系统中,订单创建后可以通过RocketMQ将订单信息发送到支付系统进行支付处理。

订单系统发送订单信息

订单系统将订单创建消息发送到RocketMQ。

import org.apache.rocketmq.client.producer.DefaultMQProducer;
import org.apache.rocketmq.client.producer.SendResult;
import org.apache.rocketmq.common.message.Message;

public class OrderProducer {
    public static void main(String[] args) throws Exception {
        DefaultMQProducer producer = new DefaultMQProducer("orderProducer");
        producer.setNamesrvAddr("localhost:9876");
        producer.start();
        String orderId = "123456";
        Message msg = new Message("orderTopic",
                "TagA",
                ("New order created: " + orderId).getBytes(RemotingHelper.DEFAULT_CHARSET));
        SendResult sendResult = producer.send(msg);
        System.out.println("Order message sent: " + sendResult);
        producer.shutdown();
    }
}

支付处理系统接收订单信息

支付处理系统收到订单信息后,进行相应的支付处理。

import org.apache.rocketmq.client.consumer.DefaultMQPushConsumer;
import org.apache.rocketmq.client.consumer.listener.MessageListenerOrderly;
import org.apache.rocketmq.common.consumer.ConsumeFromWhere;
import org.apache.rocketmq.common.message.MessageExt;

public class PaymentConsumer {
    public static void main(String[] args) throws Exception {
        DefaultMQPushConsumer consumer = new DefaultMQPushConsumer("paymentConsumer");
        consumer.setNamesrvAddr("localhost:9876");
        consumer.subscribe("orderTopic", "TagA");
        consumer.setConsumeFromWhere(ConsumeFromWhere.CONSUME_FROM_LAST_OFFSET);
        consumer.registerMessageListener((msgs, context) -> {
            for (MessageExt msg : msgs) {
                System.out.printf("New order received: %s %n", msg);
                // 进行支付处理
                pay(msg);
            }
            return ConsumeOrderlyStatus.SUCCESS;
        });
        consumer.start();
        while (true) {
            Thread.sleep(10000);
        }
    }

    private static void pay(MessageExt msg) {
        // 模拟支付处理
        System.out.println("Processing payment for order: " + new String(msg.getBody()));
    }
}

通过以上示例,我们可以看到RocketMQ在分布式场景下的应用。在实际项目中,可以结合自己的业务逻辑进行灵活配置和使用。

打开App,阅读手记
0人推荐
发表评论
随时随地看视频慕课网APP