一文详解RocketMQ-Spring的源码解析与实战

摘要:这篇文章主要介绍 Spring Boot 项目使用 rocketmq-spring SDK 实现消息收发的操作流程,同时笔者会从开发者的角度解读 SDK 的设计逻辑。

本文分享自华为云社区《RocketMQ-Spring : 实战与源码解析一网打尽》,作者:勇哥java实战分享。

RocketMQ 是大家耳熟能详的消息队列,开源项目 rocketmq-spring 可以帮助开发者在 Spring Boot 项目中快速整合 RocketMQ。

这篇文章会介绍 Spring Boot 项目使用 rocketmq-spring SDK 实现消息收发的操作流程,同时笔者会从开发者的角度解读 SDK 的设计逻辑。

一文详解RocketMQ-Spring的源码解析与实战

一 SDK 简介

一文详解RocketMQ-Spring的源码解析与实战

项目地址:https://github.com/apache/rocketmq-spring

rocketmq-spring 的本质是一个 Spring Boot starter 。

Spring Boot 基于“约定大于配置”(Convention over configuration)这一理念来快速地开发、测试、运行和部署 Spring 应用,并能通过简单地与各种启动器(如 spring-boot-web-starter)结合,让应用直接以命令行的方式运行,不需再部署到独立容器中。

Spring Boot starter 构造的启动器使用起来非常方便,开发者只需要在 pom.xml 引入 starter 的依赖定义,在配置文件中编写约定的配置即可。

下面我们看下 rocketmq-spring-boot-starter 的配置:

1、引入依赖

<dependency>
 <groupId>org.apache.rocketmq</groupId>
 <artifactId>rocketmq-spring-boot-starter</artifactId>
 <version>2.2.3</version>
</dependency>

2、约定配置

一文详解RocketMQ-Spring的源码解析与实战

接下来,我们分别按照生产者和消费者的顺序,详细的讲解消息收发的操作过程。

二 生产者

首先我们添加依赖后,进行如下三个步骤:

1、配置文件中配置如下

rocketmq:
  name-server: 127.0.0.1:9876
  producer:
      group: platform-sms-server-group
    # access-key: myaccesskey
    # secret-key: mysecretkey
  topic: sms-common-topic

生产者配置非常简单,主要配置名字服务地址和生产者组。

2、需要发送消息的类中注入 RcoketMQTemplate

@Autowired
private RocketMQTemplate rocketMQTemplate;
​
@Value("${rocketmq.topic}")
private String smsTopic;

3、发送消息,消息体可以是自定义对象,也可以是 Message 对象

rocketMQTemplate 类包含多钟发送消息的方法:

  1. 同步发送 syncSend
  2. 异步发送 asyncSend
  3. 顺序发送 syncSendOrderly
  4. oneway发送 sendOneWay

下面的代码展示如何同步发送消息。

String destination = StringUtils.isBlank(tags) ? topic : topic + ":" + tags;
SendResult sendResult =
 rocketMQTemplate.syncSend(
            destination, 
 MessageBuilder.withPayload(messageContent).
 setHeader(MessageConst.PROPERTY_KEYS, uniqueId).
 build()
 );
if (sendResult != null) {
 if (sendResult.getSendStatus() == SendStatus.SEND_OK) {
 // send message success ,do something 
 }
}

syncSend 方法的第一个参数是发送的目标,格式是:topic + ":" + tags ,

第二个参数是:spring-message 规范的 message 对象 ,而 MessageBuilder 是一个工具类,方法链式调用创建消息对象。

三 消费者

1、配置文件中配置如下

rocketmq:
  name-server: 127.0.0.1:9876
  consumer1:
    group: platform-sms-worker-common-group
    topic: sms-common-topic

2、实现消息监听器

@Component
@RocketMQMessageListener(
 consumerGroup = "${rocketmq.consumer1.group}", //消费组
    topic = "${rocketmq.consumer1.topic}" //主题
)
public class SmsMessageCommonConsumer implements RocketMQListener<String> {
 public void onMessage(String message) {
 System.out.println("普通短信:" + message);
 }
}

消费者实现类也可以实现 RocketMQListener<MessageExt>, 在 onMessage 方法里通过 RocketMQ 原生消息对象 MessageExt 获取更详细的消息数据 。

public void onMessage(MessageExt message) {
 try {
        String body = new String(message.getBody(), "UTF-8");
        logger.info("普通短信:" + message);
 } catch (Exception e) {
 logger.error("common onMessage error:", e);
 }
}

四 源码概览

一文详解RocketMQ-Spring的源码解析与实战

最新源码中,我们可以看到源码中包含四个模块:

1、rocketmq-spring-boot-parent

该模块是父模块,定义项目所有依赖的 jar 包。

2、rocketmq-spring-boot

核心模块,实现了 starter 的核心逻辑。

3、rocketmq-spring-boot-starter

SDK 模块,简单封装,外部项目引用。

4、rocketmq-spring-boot-samples

示例代码模块。这个模块非常重要,当用户使用 SDK 时,可以参考示例快速开发。

五 starter 实现

我们重点分析下 rocketmq-spring-boot 模块的核心源码:

一文详解RocketMQ-Spring的源码解析与实战

spring-boot-starter 实现需要包含如下三个部分:

1、定义 Spring 自身的依赖包和 RocketMQ 的依赖包 ;

2、定义spring.factories 文件

在 resources 包下创建 META-INF 目录后,新建 spring.factories 文件,并在文件中定义自动加载类,文件内容是:

org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
org.apache.rocketmq.spring.autoconfigure.RocketMQAutoConfiguration

spring boot 会根据文件中配置的自动化配置类来自动初始化相关的 Bean、Component 或 Service。

3、实现自动加载类

在 RocketMQAutoConfiguration 类的具体实现中,我们重点分析下生产者和消费者是如何分别启动的。

▍生产者发送模板类:RocketMQTemplate

RocketMQAutoConfiguration 类定义了两个默认的 Bean :

一文详解RocketMQ-Spring的源码解析与实战一文详解RocketMQ-Spring的源码解析与实战

首先SpringBoot项目中配置文件中的配置值会根据属性条件绑定到 RocketMQProperties 对象 中,然后使用 RocketMQ 的原生 API 分别创建生产者 Bean 和拉取消费者 Bean , 分别将两个 bean 设置到 RocketMQTemplate 对象中。

两个重点需要强调:

  • 发送消息时,将 spring-message 规范下的消息对象封装成 RocketMQ 消息对象

一文详解RocketMQ-Spring的源码解析与实战

  • 默认拉取消费者 litePullConsumer 。拉取消费者一般用于大数据批量处理场景 。

一文详解RocketMQ-Spring的源码解析与实战

RocketMQTemplate 类封装了拉取消费者的receive方法,以方便开发者使用。

一文详解RocketMQ-Spring的源码解析与实战

▍自定义消费者类

下图是并发消费者的例子:

一文详解RocketMQ-Spring的源码解析与实战

那么 rocketmq-spring 是如何自动启动消费者呢 ?

一文详解RocketMQ-Spring的源码解析与实战

spring 容器首先注册了消息监听器后置处理器,然后调用 ListenerContainerConfiguration 类的 registerContainer 方法 。

对比并发消费者的例子,我们可以看到: DefaultRocketMQListenerContainer 是对 DefaultMQPushConsumer 消费逻辑的封装。

一文详解RocketMQ-Spring的源码解析与实战

封装消费消息的逻辑,同时满足 RocketMQListener 泛化接口支持不同参数,比如 String 、MessageExt 、自定义对象 。

首先DefaultRocketMQListenerContainer初始化之后, 获取 onMessage 方法的参数类型 。

一文详解RocketMQ-Spring的源码解析与实战

然后消费者调用 consumeMessage 处理消息时,封装了一个 handleMessage 方法 ,将原生 RocketMQ 消息对象 MessageExt 转换成 onMessage 方法定义的参数对象,然后调用 rocketMQListener 的 onMessage 方法。

一文详解RocketMQ-Spring的源码解析与实战

上图右侧标红的代码也就是该方法的精髓:

rocketMQListener.onMessage(doConvertMessage(messageExt));

六 写到最后

开源项目 rocketmq-spring 有很多值得学习的地方 ,我们可以从如下四个层面逐层进阶:

1、学会如何使用 :参考 rocketmq-spring-boot-samples 模块的示例代码,学会如何发送和接收消息,快速编码;

2、模块设计:学习项目的模块分层 (父模块、SDK 模块、核心实现模块、示例代码模块);

3、starter 设计思路 :定义自动配置文件 spring.factories 、设计配置属性类 、在 RocketMQ client 的基础上实现优雅的封装、深入理解 RocketMQ 源码等;

4、举一反三:当我们理解了 rocketmq-spring 的源码,我们可以尝试模仿该项目写一个简单的 spring boot starter。

 

点击关注,第一时间了解华为云新鲜技术~

原文链接:https://www.cnblogs.com/huaweiyun/p/17349393.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:一文详解RocketMQ-Spring的源码解析与实战 - Python技术站

(0)
上一篇 2023年4月25日
下一篇 2023年4月25日

相关文章

  • Java深入理解代码块的使用细节

    Java 深入理解代码块的使用细节 代码块的定义 代码块是指被一对大括号包含起来的代码段,其中包括了定义变量、方法、循环、分支等语句。 Java中的代码块可以分为以下两种: 实例代码块 实例代码块是定义在类中的非静态代码块,可以用于初始化实例变量。实例代码块会在构造方法执行前执行。 实例代码块的示例代码如下: public class Demo { priv…

    Java 2023年5月20日
    00
  • 使用Java编写一个简单的Web的监控系统

    使用Java编写一个简单的Web监控系统需要以下几个步骤: 选择合适的监控框架:选择一个合适的监控框架来实现Web的监控,比如可以选择Spring Boot Actuator、Micrometer Actuator等。这些框架已经内置了一些用于监控Web应用程序的功能,包括HTTP请求记录、应用程序指标收集等等。 设置监控端点:在监控框架中配置监控端点,使得…

    Java 2023年5月19日
    00
  • Android编程之匿名内部类与回调函数用法分析

    Android编程之匿名内部类与回调函数用法分析 什么是匿名内部类 匿名内部类是一种没有类名的内部类,直接使用new来创建,并且实现了某个接口或者继承了某个类。使用场景通常是在需要实现较为简单的接口或者重写某个类中的方法时使用,避免了创建过多的类文件。 如何使用匿名内部类实现回调函数 在Android编程中,回调函数通常用于实现异步处理,将处理结果返回给调用…

    Java 2023年5月26日
    00
  • Spring整合Junit的使用详解

    我来为您讲解Spring整合Junit的使用详解。 什么是Junit Junit是一个Java语言的单元测试框架,它具有简单易用、扩展性强等特点。Junit在测试驱动开发(TDD)和行为驱动开发(BDD)中使用广泛。在Spring项目中,我们经常使用Junit来对项目进行单元测试和集成测试。 如何整合Spring和Junit 添加Spring和JUnit的依…

    Java 2023年5月19日
    00
  • Sentinel实现动态配置的集群流控的方法

    Sentinel是一个分布式系统的流量控制组件,其通过提供多种限流、降级、熔断等机制来保护系统的稳定性。Sentinel可以配合Spring Cloud、Dubbo等框架使用,而且其提供了动态配置的支持,通过动态更新规则实现流量控制策略的动态调整。本文将详细讲解Sentinel实现动态配置的集群流控的方法,具体过程如下: 步骤1:搭建Sentinel集群 首…

    Java 2023年6月15日
    00
  • Java的Struts框架报错“ActionServletSecurityException”的原因与解决办法

    当使用Java的Struts框架时,可能会遇到“ActionServletSecurityException”错误。这个错误通常由以下原因之一起: 安全配置错误:如果安全配置文件中没有正确配置,则可能会出现此错误。在这种情况下,需要检查文件以解决此问题。 安全限制:如果安全限制不允许访问,则可能会出现此错误。在这种情况下,需要检查安全限制以解决此问题。 以下…

    Java 2023年5月5日
    00
  • 浅谈Springboot2.0防止XSS攻击的几种方式

    浅谈Springboot2.0防止XSS攻击的几种方式 什么是XSS攻击? XSS(跨站脚本攻击),是指攻击者在web页面中插入恶意脚本,当用户浏览网页时,脚本会被执行,从而达到攻击者的目的。 常见的XSS攻击方式有反射型、存储型,以及DOM Based。在本文中,我们将围绕Springboot2.0介绍防止XSS攻击的几种方式。 1.使用HtmlUtils…

    Java 2023年5月20日
    00
  • Java实现简单连连看游戏

    Java实现简单连连看游戏攻略 界面设计 在实现连连看的过程中,一般需要自己手动设计游戏的界面。一般的要求是需要一个界面来显示游戏的进度和得分,同时还需要一个游戏区域来进行游戏。 例如,在游戏区域中,我们可以用一个二维数组来表示每个格子上的图案。我们可以使用Java Swing来实现监测用户点击的事件,并根据用户的点击来判断该元素能否进行消除,然后在界面上进…

    Java 2023年5月19日
    00
合作推广
合作推广
分享本页
返回顶部