百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术教程 > 正文

Spring Cloud Stream使用

csdh11 2024-11-30 14:15 4 浏览

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将应用程序配置为绑定接口处理器中定义的通道 INPUTOUTPUT。这两个通道都是绑定,可以配置为使用具体的消息中间件或绑定器。

让我们来看看所有这些概念的定义:

  • 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 将使用方法名称作为通道名称。因此,我们有三个通道,分别称为myInputmyOutputAnother 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/resourcesapplication.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 查询健康状态。

相关推荐

Micheal Nielsen&#39;s神经网络学习之二

依然是跟着MichaelNielsen的神经网络学习,基于前一篇的学习,已经大概明白了神经网络的基本结构和BP算法,也能通过神经网络训练数字识别功能,之后我试验了一下使用神经网络训练之前的文本分类,...

CocoaPods + XCTest进行单元测试 c单元测试工具

在使用XCTest进行单元测试时,我们经常会遇到一些CocoaPods中的开源框架的调用,比如“Realm”或“Alamofire”在测试的时候,如果配置不当,会导致“frameworknotfo...

Java基础知识回顾第四篇 java基础讲解

1、&和&&的区别作为逻辑运算符:&(不管左边是什么,右边都参与运算),&&(如果左边为false,右边则不参与运算,短路)另外&可作为位运算符...

项目中的流程及类似业务的设计模式总结

说到业务流程,可能是我做过的项目中涉及业务最多的一个方面了。除了在流程设计之外,在一些考核系统、产业审批、还有很多地方,都用到相似的设计思路,在此一并总结一下。再说到模式,并不是因为流行才用这个词,而...

联想三款显示器首批获得 Eyesafe Certified 2.0 认证

IT之家7月31日消息,据外媒报道,三款全新联想显示器是全球首批满足EyesafeCertified2.0的设备。据报道,联想获得EyesafeCertified2.0认证的显...

maven的生命周期,插件介绍(二) 一个典型的maven构建生命周期

1.maven生命周期一个完整的项目构建过程通常包括清理、编译、测试、打包、集成测试、验证、部署等步骤,Maven从中抽取了一套完善的、易扩展的生命周期。Maven的生命周期是抽象的,其中的具体任务都...

多线程(3)-基于Object的线程等待与唤醒

概述在使用synchronized进行线程同步中介绍了依赖对象锁定线程,本篇文章介绍如何依赖对象协调线程。同synchronized悲观锁一样,线程本身不能等待与唤醒,也是需要对象才能完成等待与唤醒的...

jquery mobile + 百度地图 + phonegap 写的一个&quot;校园助手&quot;的app

1jquerymobile+百度地图+phonegap写的一个"校园助手"的app,使用的是基于Flat-UI的jQueryMobile,请参考:https://github.com/...

Apache 服务启动不了 apache系统服务启动不了

{我是新手,从未遇到此问题,请各位大大勿喷}事由:今天早上上班突然发现公司网站出现问题。经过排查,发现是Apache出现问题。首先检查配置文件没有出问题后,启动服务发现Apache服务能启动,但是没法...

健康债和技术债都不能欠 公众号: 我是攻城师(woshigcs)

在Solr4.4之后,Solr提供了SolrCloud分布式集群的模式,它带来的主要好处是:(1)大数据量下更高的性能(2)更好扩展性(3)更高的可靠性(4)更简单易用什么时候应该使用Sol...

Eye Experience怎么用?HTC告诉你 eyebeam怎么用

IT之家(www.ithome.com):EyeExperience怎么用?HTC告诉你HTC上周除了发布HTCDesireEYE自拍机和HTCRE管状运动相机之外,还发布了一系列新的智能手机...

Android系统应用隐藏和应用禁止卸载

1、应用隐藏与禁用Android设置中的应用管理器提供了一个功能,就是【应用停用】功能,这是针对某些系统应用的。当应用停用之后,应用的图标会被隐藏,但apk还是存在,不会删除,核心接口就是Packag...

计算机软件技术分享--赠人玫瑰,手遗余香

一、Netty介绍Netty是由JBOSS提供的一个java开源框架。Netty提供异步的、事件驱动的网络应用程序框架和工具,用以快速开发高性能、高可靠性的网络服务器和客户端程序。也就是说,Netty...

Gecco爬虫框架的线程和队列模型 爬虫通用框架

简述爬虫在抓取一个页面后一般有两个任务,一个是解析页面内容,一个是将需要继续抓取的url放入队列继续抓取。因此,当爬取的网页很多的情况下,待抓取url的管理也是爬虫框架需要解决的问题。本文主要说的是g...

一点感悟(一) 初识 初读感知的意思

时间过得很快,在IT业已从业了两年多。人这一辈子到底需要什么,在路边看着人来人往,大部分人脸上都是很匆忙。上海真是一个魔都,它有魅力,有底蕴,但是一个外地人在这里扎根置业,真的是举全家之力,还贷3...