Spring Cloud Stream使用
csdh11 2024-11-30 14:15 24 浏览
1. 概述
Spring Cloud Stream是一个建立在Spring Boot和Spring Integration之上的框架,有助于创建事件驱动或消息驱动的微服务。
2. Maven 依赖项
首先,需要将 Spring Cloud Starter Stream 与代理 RabbitMQ Maven 依赖项作为消息中间件添加到我们的 pom.xml:
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-stream-rabbit</artifactId>
<version>3.1.3</version>
</dependency>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-stream-test-support</artifactId>
<version>3.1.3</version>
<scope>test</scope>
</dependency>
3. 主要概念
微服务架构遵循“smart endpoints and dumb pipes”原则。端点之间的通信由消息中间件方(如RabbitMQ或Apache Kafka)驱动。服务通过这些终结点或通道发布域事件进行通信。
让我们来看看构成 Spring Cloud Stream 框架的概念,以及构建消息驱动服务时必须了解的基本范式。
3.1. 构造
让我们看一下 Spring Cloud Stream中的一个简单的服务,它侦听输入绑定并向输出绑定发送响应:
@SpringBootApplication
@EnableBinding(Processor.class)
public class MyLoggerServiceApplication {
public static void main(String[] args) {
SpringApplication.run(MyLoggerServiceApplication.class, args);
}
@StreamListener(Processor.INPUT)
@SendTo(Processor.OUTPUT)
public LogMessage enrichLogMessage(LogMessage log) {
return new LogMessage(String.format("[1]: %s", log.getMessage()));
}
}
注解@EnableBinding将应用程序配置为绑定接口处理器中定义的通道 INPUT 和 OUTPUT。这两个通道都是绑定,可以配置为使用具体的消息中间件或绑定器。
让我们来看看所有这些概念的定义:
- Bindings— 以声明方式标识输入和输出通道的接口集合
- Binder — 消息中间件实现,如 Kafka 或 RabbitMQ
- Channel— 表示消息中间件和应用程序之间的通信管道
- StreamListeners — Bean 中的消息处理方法,MessageConverter 在特定于中间件的事件和域对象类型/POJO 之间进行序列化/反序列化后,将自动调用来自通道的消息
- Message Schemas — 用于消息的序列化和反序列化,这些模式可以从某个位置静态读取或动态加载,支持域对象类型的演变
3.2. 通信模式
指定到目标的消息由发布-订阅消息传递模式传递。发布者将邮件分类为主题,每个主题由名称标识。订阅者对一个或多个主题表示兴趣。中间件过滤消息,将有趣的主题传递给订阅者。
现在,订阅者可以分组。使用者组是一组订阅者或使用者,由组 ID 标识,其中来自主题或主题分区的消息以负载平衡的方式传递。
4. 编程模型
本节介绍构建Spring Cloud Stream应用程序的基础知识。
4.1. 功能测试
测试支持是一个绑定程序实现,允许与通道交互并检查消息。让我们向上面的 enrichLogMessage 服务发送一条消息,并检查响应是否在消息开头包含文本 “[1]:”:
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(classes = MyLoggerServiceApplication.class)
@DirtiesContext
public class MyLoggerApplicationTests {
@Autowired
private Processor pipe;
@Autowired
private MessageCollector messageCollector;
@Test
public void whenSendMessage_thenResponseShouldUpdateText() {
pipe.input()
.send(MessageBuilder.withPayload(new LogMessage("This is my message"))
.build());
Object payload = messageCollector.forChannel(pipe.output())
.poll()
.getPayload();
assertEquals("[1]: This is my message", payload.toString());
}
}
4.2. 自定义渠道
在上面的例子中,我们使用了Spring Cloud提供的处理器接口,它只有一个输入和一个输出通道。
如果我们需要不同的东西,比如一个输入和两个输出通道,我们可以创建一个自定义处理器:
public interface MyProcessor {
String INPUT = "myInput";
@Input
SubscribableChannel myInput();
@Output("myOutput")
MessageChannel anOutput();
@Output
MessageChannel anotherOutput();
}
Spring 将为我们提供此接口的正确实现。通道名称可以使用注解进行设置,例如在@Output(“myOutput”)中。否则,Spring 将使用方法名称作为通道名称。因此,我们有三个通道,分别称为myInput,myOutput和Another Output。
现在,假设如果值小于 10,我们希望将消息路由到一个输出,如果值大于或等于 10,则路由到另一个输出:
@Autowired
private MyProcessor processor;
@StreamListener(MyProcessor.INPUT)
public void routeValues(Integer val) {
if (val < 10) {
processor.anOutput().send(message(val));
} else {
processor.anotherOutput().send(message(val));
}
}
private static final <T> Message<T> message(T val) {
return MessageBuilder.withPayload(val).build();
}
4.3. 有条件派遣
使用 @StreamListener 注解,我们还可以使用使用 SpEL 表达式定义的任何条件来过滤我们在消费者中期望的消息。
例如,我们可以使用条件调度作为将消息路由到不同输出的另一种方法:
@Autowired
private MyProcessor processor;
@StreamListener(
target = MyProcessor.INPUT,
condition = "payload < 10")
public void routeValuesToAnOutput(Integer val) {
processor.anOutput().send(message(val));
}
@StreamListener(
target = MyProcessor.INPUT,
condition = "payload >= 10")
public void routeValuesToAnotherOutput(Integer val) {
processor.anotherOutput().send(message(val));
}
此方法的唯一限制是这些方法不得返回值。
5. 设置
让我们设置将处理来自 RabbitMQ 代理的消息的应用程序。
5.1. Binder配置
可以将应用程序配置为通过 META-INF/spring.binder 使用默认的绑定器实现:
rabbit:\
org.springframework.cloud.stream.binder.rabbit.config.RabbitMessageChannelBinderConfiguration
或者可以通过包含以下依赖项将 RabbitMQ 的绑定库添加到类路径中:
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-stream-binder-rabbit</artifactId>
<version>1.3.0.RELEASE</version>
</dependency>
如果没有提供绑定器实现,Spring 将使用通道之间的直接消息通信。
5.2. RabbitMQ配置
要将第 3.1 节中的示例配置为使用 RabbitMQ binder,需要更新位于 src/main/resources 的 application.yml:
spring:
cloud:
stream:
bindings:
input:
destination: queue.log.messages
binder: local_rabbit
output:
destination: queue.pretty.log.messages
binder: local_rabbit
binders:
local_rabbit:
type: rabbit
environment:
spring:
rabbitmq:
host: <host>
port: 5672
username: <username>
password: <password>
virtual-host: /
输入绑定将使用名为 queue.log.messages 的交换,输出绑定将使用交换 queue.pretty.log.messages。两个绑定都将使用名为 local_rabbit 的绑定程序。
请注意,不需要提前创建 RabbitMQ 交换器所或队列。运行应用程序时,将自动创建两个交换。
为了测试应用程序,我们可以使用 RabbitMQ 管理站点发布消息。在交换队列.log.messages的发布消息面板中,我们需要以JSON格式输入请求。
5.3. 自定义消息转换
Spring Cloud Stream允许对特定内容类型应用消息转换。在上面的示例中,希望提供纯文本,而不是使用 JSON 格式。
为此,将使用 MessageConverter 将自定义转换应用于 LogMessage:
@SpringBootApplication
@EnableBinding(Processor.class)
public class MyLoggerServiceApplication {
//...
@Bean
public MessageConverter providesTextPlainMessageConverter() {
return new TextPlainMessageConverter();
}
//...
}
public class TextPlainMessageConverter extends AbstractMessageConverter {
public TextPlainMessageConverter() {
super(new MimeType("text", "plain"));
}
@Override
protected boolean supports(Class<?> clazz) {
return (LogMessage.class == clazz);
}
@Override
protected Object convertFromInternal(Message<?> message,
Class<?> targetClass, Object conversionHint) {
Object payload = message.getPayload();
String text = payload instanceof String
? (String) payload
: new String((byte[]) payload);
return new LogMessage(text);
}
}
应用这些更改后,返回到“发布消息”面板,如果将标题“内容类型”设置为“文本/纯文本”,将有效负载设置为“Hello World”,它应该像以前一样工作。
5.4. 消费者Groups
当运行应用程序的多个实例时,每次输入通道中有新消息时,都会通知所有订阅者。大多数情况下,我们只需要处理一次消息。Spring Cloud Stream 通过使用者组实现此行为。
若要启用此行为,每个使用者绑定都可以使用 spring.cloud.stream.bindings.<CHANNEL>.group 属性来指定组名:
spring:
cloud:
stream:
bindings:
input:
destination: queue.log.messages
binder: local_rabbit
group: logMessageConsumers
...
6. 消息驱动的微服务
在本节中,将介绍在微服务上下文中运行Spring Cloud Stream应用程序所需的所有功能。
6.1. 扩大规模
当多个应用程序正在运行时,确保数据在使用者之间正确拆分非常重要。为此,Spring Cloud Stream 提供了两个属性:
- spring.cloud.stream.instanceCount — 正在运行的应用程序数量
- spring.cloud.stream.instanceIndex — 当前应用程序的索引
例如,如果部署了上述 MyLoggerServiceApplication 应用程序的两个实例,则两个应用程序的属性 spring.cloud.stream.instanceCount 应为 2,属性 spring.cloud.stream.instanceIndex 应分别为 0 和 1。
如果按照本文所述使用 Spring 数据流部署 Spring Cloud Stream 应用程序,则会自动设置这些属性。
6.2. 分区
域事件可以是分区消息。当我们扩展存储和提高应用程序性能时,这会有所帮助。
域事件通常具有分区键,以便它最终与相关消息位于同一分区中。
假设我们希望日志消息按消息中的第一个字母(即分区键)进行分区,并分组为两个分区。
以 A-M 开头的日志消息将有一个分区,N-Z 将有一个分区。这可以使用两个属性进行配置:
- spring.cloud.stream.bindings.output.producer.partitionKeyExpression — 用于对有效负载进行分区的表达式
- spring.cloud.stream.bindings.output.producer.partitionCount — 组的数量
有时要分区的表达式太复杂了,无法只用一行编写。对于这些情况,我们可以使用属性spring.cloud.stream.bindings.output.producer.partitionKeyExtractorClass编写自定义分区策略。
6.3. 健康指标
在微服务上下文中,还需要检测服务何时关闭或开始失败。Spring Cloud Stream 提供属性管理.health.binders.enabled 以启用活页夹的健康指示器。运行应用程序时,我们可以在 http://<host>:<port>/health 查询健康状态。
- 上一篇:深入理解SpringJDBC的解决方案
- 下一篇:Vue 3.3 正式发布
相关推荐
- SpringBoot+LayUI后台管理系统开发脚手架
-
源码获取方式:关注,转发之后私信回复【源码】即可免费获取到!项目简介本项目本着避免重复造轮子的原则,建立一套快速开发JavaWEB项目(springboot-mini),能满足大部分后台管理系统基础开...
- Spring Boot+Vue全栈开发实战,中文版高清PDF资源
-
SpringBoot+Vue全栈开发实战,中文高清PDF资源,需要的可以私我:)SpringBoot致力于简化开发配置并为企业级开发提供一系列非业务性功能,而Vue则采用数据驱动视图的方式将程序...
- 2021年超详细的java学习路线总结—纯干货分享
-
本文整理了java开发的学习路线和相关的学习资源,非常适合零基础入门java的同学,希望大家在学习的时候,能够节省时间。纯干货,良心推荐!第一阶段:Java基础...
- 探秘Spring Cache:让Java应用飞起来的秘密武器
-
探秘SpringCache:让Java应用飞起来的秘密武器在当今快节奏的软件开发环境中,性能优化显得尤为重要。SpringCache作为Spring框架的一部分,为我们提供了强大的缓存管理能力,让...
- 3,从零开始搭建SSHM开发框架(集成Spring MVC)
-
目录本专题博客已共享在(这个可能会更新的稍微一些)https://code.csdn.net/yangwei19680827/maven_sshm_blog...
- Spring Boot中如何使用缓存?超简单
-
SpringBoot中的缓存可以减少从数据库重复获取数据或执行昂贵计算的需要,从而显著提高应用程序的性能。SpringBoot提供了与各种缓存提供程序的集成,您可以在应用程序中轻松配置和使用缓...
- 我敢保证,全网没有再比这更详细的Java知识点总结了,送你啊
-
接下来你看到的将是全网最详细的Java知识点总结,全文分为三大部分:Java基础、Java框架、Java+云数据小编将为大家仔细讲解每大部分里面的详细知识点,别眨眼,从小白到大佬、零基础到精通,你绝...
- 1,从零开始搭建SSHM开发框架(环境准备)
-
目录本专题博客已共享在https://code.csdn.net/yangwei19680827/maven_sshm_blog1,从零开始搭建SSHM开发框架(环境准备)...
- 做一个适合二次开发的低代码平台,把程序员从curd中解脱出来-1
-
干程序员也有好长时间了,大多数时间都是在做curd。现在想做一个通用的curd平台直接将我们解放出来;把核心放在业务处理中。用过代码生成器,在数据表设计好之后使用它就可以生成需要的controller...
- 设计一个高性能Java Web框架(java做网站的框架)
-
设计一个高性能JavaWeb框架在当今互联网高速发展的时代,构建高性能的JavaWeb框架对于提升用户体验至关重要。本文将从多个角度探讨如何设计这样一个框架,让我们一起进入这段充满挑战和乐趣的旅程...
- 【推荐】强&牛!一款开源免费的功能强大的代码生成器系统!
-
今天,给大家推荐一个代码生成器系统项目,这个项目目前收获了5.3KStar,个人觉得不错,值得拿出来和大家分享下。这是我目前见过最好的代码生成器系统项目。功能完整,代码结构清晰。...
- Java面试题及答案总结(2025版持续更新)
-
大家好,我是Java面试分享最近很多小伙伴在忙着找工作,给大家整理了一份非常全面的Java面试场景题及答案。...
- Java开发网站架构演变过程-从单体应用到微服务架构详解
-
Java开发网站架构演变过程,到目前为止,大致分为5个阶段,分别为单体架构、集群架构、分布式架构、SOA架构和微服务架构。下面玄武老师来给大家详细介绍下这5种架构模式的发展背景、各自优缺点以及涉及到的...
- 本地缓存GuavaCache(一)(guava本地缓存原理)
-
在并发量、吞吐量越来越大的情况下往往是离不开缓存的,使用缓存能减轻数据库的压力,临时存储数据。根据不同的场景选择不同的缓存,分布式缓存有Redis,Memcached、Tair、EVCache、Aer...
- 一周热门
- 最近发表
- 标签列表
-
- mydisktest_v298 (34)
- document.appendchild (35)
- 头像打包下载 (61)
- acmecadconverter_8.52绿色版 (39)
- word文档批量处理大师破解版 (36)
- server2016安装密钥 (33)
- mysql 昨天的日期 (37)
- parsevideo (33)
- 个人网站源码 (37)
- centos7.4下载 (33)
- mysql 查询今天的数据 (34)
- intouch2014r2sp1永久授权 (36)
- 先锋影音源资2019 (35)
- jdk1.8.0_191下载 (33)
- axure9注册码 (33)
- pts/1 (33)
- spire.pdf 破解版 (35)
- shiro jwt (35)
- sklearn中文手册pdf (35)
- itextsharp使用手册 (33)
- 凯立德2012夏季版懒人包 (34)
- 冒险岛代码查询器 (34)
- 128*128png图片 (34)
- jdk1.8.0_131下载 (34)
- dos 删除目录下所有子目录及文件 (36)