Spring Cloud Stream 体系及原理介绍,我把所有Java框架整理成了PDF

}

boolean send(Message<?> message, long timeout);

}

消息通道里的消息如何被消费呢?
  • 由消息通道的子接口可订阅的消息通道SubscribableChannel实现,被MessageHandler消息处理器所订阅:

public interface SubscribableChannel extends MessageChannel {

boolean subscribe(MessageHandler handler);

boolean unsubscribe(MessageHandler handler);

}

  • 由MessageHandler真正地消费/处理消息:

@FunctionalInterface

public interface MessageHandler {

void handleMessage(Message<?> message) throws MessagingException;

}

Spring Messaging 内部在消息模型的基础上衍生出了其它的一些功能,如:
  • 消息接收参数及返回值处理:消息接收参数处理器HandlerMethodArgumentResolver配合@Header, @Payload等注解使用;消息接收后的返回值处理器HandlerMethodReturnValueHandler配合@SendTo注解使用;

  • 消息体内容转换器MessageConverter;

  • 统一抽象的消息发送模板AbstractMessageSendingTemplate;

  • 消息通道拦截器ChannelInterceptor;

Spring Integration

=======================================================================================

Spring Integration 提供了 Spring 编程模型的扩展用来支持企业集成模式(Enterprise Integration Patterns),是对 Spring Messaging 的扩展。

它提出了不少新的概念,包括消息路由MessageRoute、消息分发MessageDispatcher、消息过滤Filter、消息转换Transformer、消息聚合Aggregator、消息分割Splitter等等。同时还提供了MessageChannel和MessageHandler的实现,分别包括 DirectChannel、ExecutorChannel、PublishSubscribeChannel和MessageFilter、ServiceActivatingHandler、MethodInvokingSplitter 等内容。

这里为大家介绍几种消息的处理方式:
  • 消息的分割:

4.png

  • 消息的聚合:

5.png

  • 消息的过滤:

6.png

  • 消息的分发:

7.png

接下来,我们以一个最简单的例子来尝试一下 Spring Integration。

这段代码解释为:

SubscribableChannel messageChannel =new DirectChannel(); // 1

messageChannel.subscribe(msg-> { // 2

System.out.println("receive: " +msg.getPayload());

});

messageChannel.send(MessageBuilder.withPayload(“msgfrom alibaba”).build()); // 3

  • 构造一个可订阅的消息通道messageChannel。

  • 使用MessageHandler去消费这个消息通道里的消息。

  • 发送一条消息到这个消息通道,消息最终被消息通道里的MessageHandler所消费。

  • 最后控制台打印出:receive: msg from alibaba。

DirectChannel内部有个UnicastingDispatcher类型的消息分发器,会分发到对应的消息通道MessageChannel中,从名字也可以看出来,UnicastingDispatcher是个单播的分发器,只能选择一个消息通道。那么如何选择呢? 内部提供了LoadBalancingStrategy负载均衡策略,默认只有轮询的实现,可以进行扩展。

我们对上段代码做一点修改,使用多个 MessageHandler 去处理消息:

SubscribableChannel messageChannel = new DirectChannel();

messageChannel.subscribe(msg -> {

System.out.println("receive1: " + msg.getPayload());

});

messageChannel.subscribe(msg -> {

System.out.println("receive2: " + msg.getPayload());

});

messageChannel.send(MessageBuilder.withPayload(“msg from alibaba”).build());

messageChannel.send(MessageBuilder.withPayload(“msg from alibaba”).build());

由于DirectChannel内部的消息分发器是UnicastingDispatcher单播的方式,并且采用轮询的负载均衡策略,所以这里两次的消费分别对应这两个MessageHandler。控制台打印出:

receive1: msg from alibaba

receive2: msg from alibaba

既然存在单播的消息分发器UnicastingDispatcher,必然也会存在广播的消息分发器,那就是BroadcastingDispatcher,它被 PublishSubscribeChannel 这个消息通道所使用。广播消息分发器会把消息分发给所有的 MessageHandler:

SubscribableChannel messageChannel = new PublishSubscribeChannel();

messageChannel.subscribe(msg -> {

System.out.println("receive1: " + msg.getPayload());

});

messageChannel.subscribe(msg -> {

System.out.println("receive2: " + msg.getPayload());

});

messageChannel.send(MessageBuilder.withPayload(“msg from alibaba”).build());

messageChannel.send(MessageBuilder.withPayload(“msg from alibaba”).build());

Spring Cloud Stream

========================================================================================

SCS 与各模块之间的关系是:
  • SCS 在 Spring Integration 的基础上进行了封装,提出了Binder, Binding, @EnableBinding, @StreamListener等概念。

  • SCS 与 Spring Boot Actuator 整合,提供了/bindings, /channelsendpoint。

  • SCS 与 Spring Boot Externalized Configuration 整合,提供了BindingProperties, BinderProperties等外部化配置类。

  • SCS 增强了消息发送失败的和消费失败情况下的处理逻辑等功能。

  • SCS 是 Spring Integration 的加强,同时与 Spring Boot 体系进行了融合,也是 Spring Cloud Bus 的基础。它屏蔽了底层消息中间件的实现细节,希望以统一的一套 API 来进行消息的发送/消费,底层消息中间件的实现细节由各消息中间件的 Binder 完成。

Binder是提供与外部消息中间件集成的组件,为构造Binding提供了 2 个方法,分别是bindConsumer和bindProducer,它们分别用于构造生产者和消费者。目前官方的实现有 Rabbit Binder 和 Ka

【一线大厂Java面试题解析+后端开发学习笔记+最新架构讲解视频+实战项目源码讲义】

开源完整内容戳这里

fka Binder, Spring Cloud Alibaba 内部已经实现了 RocketMQ Binder。

8.png

从图中可以看出,Binding是连接应用程序跟消息中间件的桥梁,用于消息的消费和生产。我们来看一个最简单的使用 RocketMQ Binder 的例子,然后分析一下它的底层处理原理:

  • 启动类及消息的发送:

@SpringBootApplication

@EnableBinding({ Source.class, Sink.class }) // 1

public class SendAndReceiveApplication {

public static void main(String[] args) {

SpringApplication.run(SendAndReceiveApplication.class, args);

}

@Bean // 2

public CustomRunner customRunner() {

return new CustomRunner();

}

public static class CustomRunner implements CommandLineRunner {

@Autowired

private Source source;

@Override

public void run(String… args) throws Exception {

int count = 5;

for (int index = 1; index <= count; index++) {

source.output().send(MessageBuilder.withPayload(“msg-” + index).build()); // 3

}

}

}

}

  • 消息的接收:

@Service

public class StreamListenerReceiveService {

@StreamListener(Sink.INPUT) // 4

public void receiveByStreamListener1(String receiveMsg) {

System.out.println("receiveByStreamListener: " + receiveMsg);

}

}

这段代码很简单,没有涉及到 RocketMQ 相关的代码,消息的发送和接收都是基于 SCS 体系完成的。如果想切换成 RabbitMQ 或 Kafka,只需修改配置文件即可,代码无需修改。

我们来分析下这段代码的原理:

1.@EnableBinding对应的两个接口属性Source和Sink是 SCS 内部提供的。SCS 内部会基于Source和Sink构造BindableProxyFactory,且对应的 output 和 input 方法返回的 MessageChannel 是DirectChannel。output 和 input 方法修饰的注解对应的 value 是配置文件中 binding 的 name。

public interface Source {

String OUTPUT = “output”;

@Output(Source.OUTPUT)

MessageChannel output();

}

public interface Sink {

String INPUT = “input”;

@Input(Sink.INPUT)

SubscribableChannel input();

}

配置文件里 bindings 的 name 为 output 和 input,对应Source和Sink接口的方法上的注解里的 value:

spring.cloud.stream.bindings.output.destination=test-topic

spring.cloud.stream.bindings.output.content-type=text/plain

spring.cloud.stream.rocketmq.bindings.output.producer.group=demo-group

spring.cloud.stream.bindings.input.destination=test-topic

spring.cloud.stream.bindings.input.content-type=text/plain

spring.cloud.stream.bindings.input.group=test-group1

  1. 构造CommandLineRunner,程序启动的时候会执行CustomRunner的run方法。

  2. 调用Source接口里的 output 方法获取DirectChannel,并发送消息到这个消息通道中。这里跟之前 Spring Integration 章节里的代码一致。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值