https://docs.spring.io/spring-boot/docs/3.2.0/reference/htmlsingle/#messaging.rsocket
RSocket是一个用于字节流传输的二进制协议。它通过单个连接上的异步消息传递启用对称的交互模型。
Spring框架的spring-messaging
模块为客户端和服务器端的RSocket请求者和响应者提供支持。
RSocket策略自动配置
Spring Boot会自动配置一个RSocketStrategies
bean,该bean为RSocket有效载荷的编码和解码提供了所有必要的基础设施。默认情况下,自动配置将尝试按顺序配置以下组件:
- 使用Jackson的CBOR编解码器
- 使用Jackson的JSON编解码器
spring-boot-starter-rsocket
启动器提供了这两个依赖项。
开发人员可以通过创建实现RSocketStrategiesCustomizer
接口的bean来定制RSocketStrategies
组件。请注意,它们的@Order
很重要,因为它决定了编解码器的顺序。
RSocket服务器自动配置
Spring Boot 提供了 RSocket 服务器的自动配置功能。所需的依赖项由 spring-boot-starter-rsocket
提供。
Spring Boot 允许从 WebFlux 服务器通过 WebSocket 公开 RSocket,或者建立独立的 RSocket 服务器。这取决于应用程序的类型及其配置。
对于 WebFlux 应用程序(即类型为 WebApplicationType.REACTIVE
的应用程序),仅当以下属性匹配时,RSocket 服务器才会插入到 Web 服务器中:
spring.rsocket.server.mapping-path=/rsocket
spring.rsocket.server.transport=websocket
注意:将 RSocket 集成到 Web 服务器仅支持使用 Reactor Netty,因为 RSocket 本身是使用该库构建的。
作为替代方案,可以启动一个独立的、内嵌式的 RSocket TCP 或 websocket 服务器。除了依赖项要求外,唯一必需的配置是定义该服务器的端口:
spring.rsocket.server.port=9898
Spring Messaging RSocket 支持
Spring Boot 会自动配置 RSocket 的 Spring Messaging 基础设施。
这意味着 Spring Boot 将创建一个 RSocketMessageHandler
bean,该 bean 将处理发送到你的应用程序的 RSocket 请求。
使用 RSocketRequester 调用 RSocket 服务
一旦服务器和客户端之间建立了 RSocket 通道,任何一方都可以向另一方发送或接收请求。
作为服务器,可以在 RSocket @Controller
的任何处理程序方法上注入一个 RSocketRequester
实例。作为客户端,首先需要配置和建立 RSocket 连接。Spring Boot 为此类情况自动配置了带有预期编解码器的 RSocketRequester.Builder
,并应用了任何 RSocketConnectorConfigurer
bean。
RSocketRequester.Builder
实例是一个原型 bean,这意味着每个注入点都会为你提供一个新实例。这是有意的,因为这个构建器是有状态的,不应该使用同一个实例创建具有不同设置的请求者。
以下代码展示了一个典型的例子:
import reactor.core.publisher.Mono;
import org.springframework.messaging.rsocket.RSocketRequester;
import org.springframework.stereotype.Service;
@Service
public class MyService {
private final RSocketRequester rsocketRequester;
public MyService(RSocketRequester.Builder rsocketRequesterBuilder) {
this.rsocketRequester = rsocketRequesterBuilder.tcp("example.org", 9898);
}
public Mono<User> someRSocketCall(String name) {
return this.rsocketRequester.route("user").data(name).retrieveMono(User.class);
}
}