rsocket-java 高效的服务间通讯

参考:
rsocket 官方文档
rsocket-java demo示例 - request/response模式下的路由请求
SpringBoot整合RSocket实时数据通信
spring官方文档介绍rsocket
集成spring-rsocket
spring官方集成rsocket高级配置

一、背景

大型分布式系统通常由不同的团队使用各种技术和编程语言以模块化方式实现。这些部件需要可靠地通信,并支持快速、独立的演进。在分布式系统中,模块之间有效且可扩展的通信是一个至关重要的问题。它显著影响用户体验的延迟程度以及构建和运行系统所需的资源量。

二、交互模型

Fire-and-Forget

即发即弃是一种请求/响应的优化,在不需要响应时非常有用。它可以实现显着的性能优化,不仅可以通过跳过响应来节省网络使用量,还可以节省客户端和服务器的处理时间,因为不需要记录来等待和关联响应或取消请求。

此交互模型对于支持有损的用例非常有用,例如非关键事件日志记录。

java 复制代码
Future<Void> completionSignalOfSend = socketClient.fireAndForget(message);

Request/Response (single-response)

仍然支持标准请求/响应语义,并且仍然期望代表 RSocket 连接上的大多数请求。这些请求/响应交互可以被视为优化的"只有 1 个响应的流",并且是通过单个连接复用的异步消息。

消费者"等待"响应消息,因此它看起来像典型的请求/响应,但其底层从不同步阻塞。

java 复制代码
Future<Payload> response = socketClient.requestResponse(requestPayload);

Request/Stream (multi-response, finite)

从Request/Response扩展而来的是Request/Stream,它允许将多个消息流式传输回来。将此视为"集合"或"列表"响应,但不是将所有数据作为单个响应返回,而是每个元素按顺序流式传输回来。

使用情景可能包括:

  • 获取视频列表
  • 获取目录中的产品
  • 逐行检索文件
java 复制代码
Publisher<Payload> response = socketClient.requestStream(requestPayload);

Channel

通道是双向的,在两个方向上都有消息流

常用于此交互模型的示例用例是:

  • 客户端请求一个数据流,该数据流最初会打破当前的世界观
  • 当发生变化时,增量/差异从服务器发送到客户端
  • 客户端随着时间的推移更新订阅以添加/删除标准/主题/等。

如果没有双向通道,客户端将不得不取消初始请求、重新请求并从头开​​始接收所有数据,而不是仅仅更新订阅并有效地获取差异。

java 复制代码
Publisher<Payload> output = socketClient.requestChannel(Publisher<Payload> input);

三、示例

Request/Response (single-response)

包含路由模式,通过指定不同的路由地址,自定义实现不同的服务处理逻辑

server:

java 复制代码
RSocketServer.create(
    SocketAcceptor.forRequestResponse(
        payload -> {
            final String route = decodeRoute(payload.sliceMetadata());

            log.info("接收来自客户端的路由请求[route={}]", route);

            payload.release();

            if ("my.test.route".equals(route)) {
            	// 返回消息给客户端
                return Mono.just(ByteBufPayload.create("Hello From My Test Route"));
            }

            return Mono.error(new IllegalArgumentException("Route " + route + " not found"));
        }))
.bindNow(TcpServerTransport.create("localhost", 7000));

client:

java 复制代码
RSocket socket =
     RSocketConnector.create()
         // here we specify that route will be encoded using
         // Routing&Tagging Metadata layout specified at this
         // subspec https://github.com/rsocket/rsocket/blob/master/Extensions/Routing.md
         .metadataMimeType(WellKnownMimeType.MESSAGE_RSOCKET_ROUTING.getString())
         .connect(TcpClientTransport.create("localhost", 7000))
         .block();

 final ByteBuf routeMetadata =
     TaggingMetadataCodec.createTaggingContent(
         ByteBufAllocator.DEFAULT, Collections.singletonList("my.test.route"));
 socket
     .requestResponse(ByteBufPayload.create(ByteBufUtil.writeUtf8(ByteBufAllocator.DEFAULT, "HelloWorld"), routeMetadata))
     .map(Payload::getDataUtf8)
     .onErrorReturn("error")
     .doOnNext(p -> System.out.println("服务器返回消息: " + p))
     .log()
     .block();
相关推荐
漫漫进阶路2 小时前
VS C++ 配置OPENCV环境
开发语言·c++·opencv
陈平安Java and C2 小时前
MyBatisPlus
java
秋野酱2 小时前
如何在 Spring Boot 中实现自定义属性
java·数据库·spring boot
安的列斯凯奇3 小时前
SpringBoot篇 单元测试 理论篇
spring boot·后端·单元测试
Bunny02123 小时前
SpringMVC笔记
java·redis·笔记
BinaryBardC3 小时前
Swift语言的网络编程
开发语言·后端·golang
feng_blog66883 小时前
【docker-1】快速入门docker
java·docker·eureka
code_shenbing3 小时前
基于 WPF 平台使用纯 C# 制作流体动画
开发语言·c#·wpf
邓熙榆3 小时前
Haskell语言的正则表达式
开发语言·后端·golang
ac-er88884 小时前
Yii框架中的队列:如何实现异步操作
android·开发语言·php