Spring Cloud Stream使用
csdh11 2024-11-30 14:15 19 浏览
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 正式发布
相关推荐
- 探索Java项目中日志系统最佳实践:从入门到精通
-
探索Java项目中日志系统最佳实践:从入门到精通在现代软件开发中,日志系统如同一位默默无闻却至关重要的管家,它记录了程序运行中的各种事件,为我们排查问题、监控性能和优化系统提供了宝贵的依据。在Java...
- 用了这么多年的java日志框架,你真的弄懂了吗?
-
在项目开发过程中,有一个必不可少的环节就是记录日志,相信只要是个程序员都用过,可是咱们自问下,用了这么多年的日志框架,你确定自己真弄懂了日志框架的来龙去脉嘛?下面笔者就详细聊聊java中常用日志框架的...
- 物理老师教你学Java语言(中篇)(物理专业学编程)
-
第四章物质的基本结构——类与对象...
- 一文搞定!Spring Boot3 定时任务操作全攻略
-
各位互联网大厂的后端开发小伙伴们,在使用SpringBoot3开发项目时,你是否遇到过定时任务实现的难题呢?比如任务调度时间不准确,代码报错却找不到方向,是不是特别头疼?如今,随着互联网业务规模...
- 你还不懂java的日志系统吗 ?(java的日志类)
-
一、背景在java的开发中,使用最多也绕不过去的一个话题就是日志,在程序中除了业务代码外,使用最多的就是打印日志。经常听到的这样一句话就是“打个日志调试下”,没错在日常的开发、调试过程中打印日志是常干...
- 谈谈枚举的新用法--java(java枚举的作用与好处)
-
问题的由来前段时间改游戏buff功能,干了一件愚蠢的事情,那就是把枚举和运算集合在一起,然后运行一段时间后buff就出现各种问题,我当时懵逼了!事情是这样的,做过游戏的都知道,buff,需要分类型,且...
- 你还不懂java的日志系统吗(javaw 日志)
-
一、背景在java的开发中,使用最多也绕不过去的一个话题就是日志,在程序中除了业务代码外,使用最多的就是打印日志。经常听到的这样一句话就是“打个日志调试下”,没错在日常的开发、调试过程中打印日志是常干...
- Java 8之后的那些新特性(三):Java System Logger
-
去年12月份log4j日志框架的一个漏洞,给Java整个行业造成了非常大的影响。这个事情也顺带把log4j这个日志框架推到了争议的最前线。在Java领域,log4j可能相对比较流行。而在log4j之外...
- Java开发中的日志管理:让程序“开口说话”
-
Java开发中的日志管理:让程序“开口说话”日志是程序员的朋友,也是程序的“嘴巴”。它能让程序在运行过程中“开口说话”,告诉我们它的状态、行为以及遇到的问题。在Java开发中,良好的日志管理不仅能帮助...
- OS X 效率启动器 Alfred 详解与使用技巧
-
问:为什么要在Mac上使用效率启动器类应用?答:在非特殊专业用户的环境下,(每天)用户一般可以在系统中进行上百次操作,可以是点击,也可以是拖拽,但这些只是过程,而我们的真正目的是想获得结果,也就是...
- Java中 高级的异常处理(java中异常处理的两种方式)
-
介绍异常处理是软件开发的一个关键方面,尤其是在Java中,这种语言以其稳健性和平台独立性而闻名。正确的异常处理不仅可以防止应用程序崩溃,还有助于调试并向用户提供有意义的反馈。...
- 【性能调优】全方位教你定位慢SQL,方法介绍下!
-
1.使用数据库自带工具...
- 全面了解mysql锁机制(InnoDB)与问题排查
-
MySQL/InnoDB的加锁,一直是一个常见的话题。例如,数据库如果有高并发请求,如何保证数据完整性?产生死锁问题如何排查并解决?下面是不同锁等级的区别表级锁:开销小,加锁快;不会出现死锁;锁定粒度...
- 看懂这篇文章,你就懂了数据库死锁产生的场景和解决方法
-
一、什么是死锁加锁(Locking)是数据库在并发访问时保证数据一致性和完整性的主要机制。任何事务都需要获得相应对象上的锁才能访问数据,读取数据的事务通常只需要获得读锁(共享锁),修改数据的事务需要获...
- 一周热门
- 最近发表
- 标签列表
-
- 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)
- 反恐24小时电话铃声 (33)
- 冒险岛代码查询器 (34)
- 128*128png图片 (34)
- jdk1.8.0_131下载 (34)