Spring WebFlux包括一个用于执行HTTP请求的客户端。 WebClient
使用函数风格的流式API,基于Reactor实现,详细内容请参见web-reactive.xml,它实现了声明式的异步逻辑组合,而无需处理线程或者并发相关内容。 它是完全非阻塞的,支持流,并且依赖于与服务端相同的编解码器,该[编解码器]也用于在服务器端对请求和响应内容进行编码和解码。
WebClient
需要一个HTTP客户端库来发出请求。内建支持以下几种:
-
Others can be plugged via
ClientHttpConnector
.
创建WebClient
最简单的方式是通过下面的静态工厂方法:
-
WebClient.create()
-
WebClient.create(String baseUrl)
也可以使用WebClient.builder()
进行相关配置的自定义:
-
uriBuilderFactory
: 自定义UriBuilderFactory
使用一个基本的URL。 -
defaultUriVariables
: 填充URI模板时使用的默认值。 -
defaultHeader
: 请求默认Header。 -
defaultCookie
: 请求默认Cookies。 -
defaultRequest
: 用来定制请求的Consumer
。 -
filter
: 处理请求的客户端过滤器。 -
exchangeStrategies
: 定制HTTP message reader/writer。 -
clientConnector
: HTTP客户端库相关配置。
例如:
Java.
WebClient client = WebClient.builder()
.codecs(configurer -> ... )
.build();
Kotlin.
val webClient = WebClient.builder()
.codecs { configurer -> ... }
.build()
一旦构建完成,WebClient
就是不可变。但是,你可以创建一个克隆实例来进行配置修改。
Java.
WebClient client1 = WebClient.builder()
.filter(filterA).filter(filterB).build();
WebClient client2 = client1.mutate()
.filter(filterC).filter(filterD).build();
// client1 has filterA, filterB
// client2 has filterA, filterB, filterC, filterD
Kotlin.
val client1 = WebClient.builder()
.filter(filterA).filter(filterB).build()
val client2 = client1.mutate()
.filter(filterC).filter(filterD).build()
// client1 has filterA, filterB
// client2 has filterA, filterB, filterC, filterD
编解码器限制了内存缓冲区的大小来避免一些内存问题。默认值是256KB. 如果不够的话,会出现以下错误:
org.springframework.core.io.buffer.DataBufferLimitException: Exceeded limit on max bytes to buffer
可以通过以下方式来修改限制:
Java.
WebClient webClient = WebClient.builder()
.codecs(configurer -> configurer.defaultCodecs().maxInMemorySize(2 * 1024 * 1024))
.build();
Kotlin.
val webClient = WebClient.builder()
.codecs { configurer -> configurer.defaultCodecs().maxInMemorySize(2 * 1024 * 1024) }
.build()
如果想定制Reactor Netty的配置,提供一个预定义的HttpClient
:
Java.
HttpClient httpClient = HttpClient.create().secure(sslSpec -> ...);
WebClient webClient = WebClient.builder()
.clientConnector(new ReactorClientHttpConnector(httpClient))
.build();
Kotlin.
val httpClient = HttpClient.create().secure { ... }
val webClient = WebClient.builder()
.clientConnector(ReactorClientHttpConnector(httpClient))
.build()
默认情况下,HttpClient
使用存储在reactor.netty.http.HttpResources
中的全局Reactor Netty资源,包括事件循环和连接池。对于事件循环并发模式来说这是推荐的,优化的使用方法。在这种使用方法下,全局资源在进程结束前一直保持活跃状态。
如果服务器与进程的生命周期相同,通常没有必要显示的去关闭。但是,如果服务器可以在进程内启动或者停止(例如,以WAR方式部署的Spring MVC应用),可以定义一个Spring管理的ReactorResourceFactory
类型(默认globalResources=true
)的bean来保证Reactor Netty全局资源在Spring ApplicationContext
关闭的时候也随之关闭,如下所示:
Java.
@Bean
public ReactorResourceFactory reactorResourceFactory() {
return new ReactorResourceFactory();
}
Kotlin.
@Bean
fun reactorResourceFactory() = ReactorResourceFactory()
你也可以选择不使用Reactor Netty全局资源。但是在这种情况下,需要你自行确保所有的Reactor Netty客户端和服务端都是用同样的共享资源,如下:
Java.
@Bean
public ReactorResourceFactory resourceFactory() {
ReactorResourceFactory factory = new ReactorResourceFactory();
factory.setUseGlobalResources(false); ①
return factory;
}
@Bean
public WebClient webClient() {
Function<HttpClient, HttpClient> mapper = client -> {
// Further customizations...
};
ClientHttpConnector connector =
new ReactorClientHttpConnector(resourceFactory(), mapper); ②
return WebClient.builder().clientConnector(connector).build(); ③
}
① 创建独立于全局的资源。
② 使用ReactorClientHttpConnector
带有资源工厂参数的构造函数。
③ 把connector注入到WebClient.Builder
。
Kotlin.
@Bean
fun resourceFactory() = ReactorResourceFactory().apply {
isUseGlobalResources = false ①
}
@Bean
fun webClient(): WebClient {
val mapper: (HttpClient) -> HttpClient = {
// Further customizations...
}
val connector = ReactorClientHttpConnector(resourceFactory(), mapper) ②
return WebClient.builder().clientConnector(connector).build() ③
}
① 创建独立于全局的资源。
② 使用ReactorClientHttpConnector
带有资源工厂参数的构造函数。
③ 把connector注入到WebClient.Builder
。
配置连接超时时间:
Java.
import io.netty.channel.ChannelOption;
HttpClient httpClient = HttpClient.create()
.option(ChannelOption.CONNECT_TIMEOUT_MILLIS, 10000);
WebClient webClient = WebClient.builder()
.clientConnector(new ReactorClientHttpConnector(httpClient))
.build();
Kotlin.
import io.netty.channel.ChannelOption
val httpClient = HttpClient.create()
.option(ChannelOption.CONNECT_TIMEOUT_MILLIS, 10000);
val webClient = WebClient.builder()
.clientConnector(new ReactorClientHttpConnector(httpClient))
.build();
配置读写超时时间:
Java.
import io.netty.handler.timeout.ReadTimeoutHandler;
import io.netty.handler.timeout.WriteTimeoutHandler;
HttpClient httpClient = HttpClient.create()
.doOnConnected(conn -> conn
.addHandlerLast(new ReadTimeoutHandler(10))
.addHandlerLast(new WriteTimeoutHandler(10)));
// Create WebClient...
Kotlin.
import io.netty.handler.timeout.ReadTimeoutHandler
import io.netty.handler.timeout.WriteTimeoutHandler
val httpClient = HttpClient.create()
.doOnConnected { conn -> conn
.addHandlerLast(new ReadTimeoutHandler(10))
.addHandlerLast(new WriteTimeoutHandler(10))
}
// Create WebClient...
为所有请求配置响应超时时间:
Java.
HttpClient httpClient = HttpClient.create()
.responseTimeout(Duration.ofSeconds(2));
// Create WebClient...
Kotlin.
val httpClient = HttpClient.create()
.responseTimeout(Duration.ofSeconds(2));
// Create WebClient...
为单个请求配置响应超时时间:
Java.
WebClient.create().get()
.uri("https://example.org/path")
.httpRequest(httpRequest -> {
HttpClientRequest reactorRequest = httpRequest.getNativeRequest();
reactorRequest.responseTimeout(Duration.ofSeconds(2));
})
.retrieve()
.bodyToMono(String.class);
Kotlin.
WebClient.create().get()
.uri("https://example.org/path")
.httpRequest { httpRequest: ClientHttpRequest ->
val reactorRequest = httpRequest.getNativeRequest<HttpClientRequest>()
reactorRequest.responseTimeout(Duration.ofSeconds(2))
}
.retrieve()
.bodyToMono(String::class.java)
以下示例展示了如何定制Jetty HttpClient
的配置:
Java.
HttpClient httpClient = new HttpClient();
httpClient.setCookieStore(...);
WebClient webClient = WebClient.builder()
.clientConnector(new JettyClientHttpConnector(httpClient))
.build();
Kotlin.
val httpClient = HttpClient()
httpClient.cookieStore = ...
val webClient = WebClient.builder()
.clientConnector(new JettyClientHttpConnector(httpClient))
.build();
默认情况下, HttpClient
创建它自己的资源 (Executor
, ByteBufferPool
, Scheduler
),直到进程推出或者 stop()
方法被调用才会销毁。
在多个 Jetty client (and server) 之间可以共享资源,并且确保在Spring ApplicationContext
关闭的时候被关闭,可以通过定义一个受Spring管理的类型 JettyResourceFactory
的Bean来实现,例如:
Java.
@Bean
public JettyResourceFactory resourceFactory() {
return new JettyResourceFactory();
}
@Bean
public WebClient webClient() {
HttpClient httpClient = new HttpClient();
// Further customizations...
ClientHttpConnector connector =
new JettyClientHttpConnector(httpClient, resourceFactory()); ①
return WebClient.builder().clientConnector(connector).build(); ②
}
① 使用带有资源工厂参数的 JettyClientHttpConnector
构造器
② 把connector配置到WebClient.Builder
Kotlin.
@Bean
fun resourceFactory() = JettyResourceFactory()
@Bean
fun webClient(): WebClient {
val httpClient = HttpClient()
// Further customizations...
val connector = JettyClientHttpConnector(httpClient, resourceFactory()) ①
return WebClient.builder().clientConnector(connector).build() ②
}
① 使用带有资源工厂参数的 JettyClientHttpConnector
构造器
② 把connector配置到WebClient.Builder
下面的例子展示了如何定制化Apache HttpComponents HttpClient
的配置:
Java.
HttpAsyncClientBuilder clientBuilder = HttpAsyncClients.custom();
clientBuilder.setDefaultRequestConfig(...);
CloseableHttpAsyncClient client = clientBuilder.build();
ClientHttpConnector connector = new HttpComponentsClientHttpConnector(client);
WebClient webClient = WebClient.builder().clientConnector(connector).build();
Kotlin.
val client = HttpAsyncClients.custom().apply {
setDefaultRequestConfig(...)
}.build()
val connector = HttpComponentsClientHttpConnector(client)
val webClient = WebClient.builder().clientConnector(connector).build()
retrieve()
方法可以用来声明如何获取响应。例如:
Java.
WebClient client = WebClient.create("https://example.org");
Mono<ResponseEntity<Person>> result = client.get()
.uri("/persons/{id}", id).accept(MediaType.APPLICATION_JSON)
.retrieve()
.toEntity(Person.class);
Kotlin.
val client = WebClient.create("https://example.org")
val result = client.get()
.uri("/persons/{id}", id).accept(MediaType.APPLICATION_JSON)
.retrieve()
.toEntity<Person>().awaitSingle()
或者仅仅获取响应体:
Java.
WebClient client = WebClient.create("https://example.org");
Mono<Person> result = client.get()
.uri("/persons/{id}", id).accept(MediaType.APPLICATION_JSON)
.retrieve()
.bodyToMono(Person.class);
Kotlin.
val client = WebClient.create("https://example.org")
val result = client.get()
.uri("/persons/{id}", id).accept(MediaType.APPLICATION_JSON)
.retrieve()
.awaitBody<Person>()
获取一个已经解码的对象流:
Java.
Flux<Quote> result = client.get()
.uri("/quotes").accept(MediaType.TEXT_EVENT_STREAM)
.retrieve()
.bodyToFlux(Quote.class);
Kotlin.
val result = client.get()
.uri("/quotes").accept(MediaType.TEXT_EVENT_STREAM)
.retrieve()
.bodyToFlow<Quote>()
默认情况下,4xx和5xx响应导致 WebClientResponseException
及其子类。可以使用 onStatus
对异常响应的处理逻辑进行自定义,例如:
Java.
Mono<Person> result = client.get()
.uri("/persons/{id}", id).accept(MediaType.APPLICATION_JSON)
.retrieve()
.onStatus(HttpStatus::is4xxClientError, response -> ...)
.onStatus(HttpStatus::is5xxServerError, response -> ...)
.bodyToMono(Person.class);
Kotlin.
val result = client.get()
.uri("/persons/{id}", id).accept(MediaType.APPLICATION_JSON)
.retrieve()
.onStatus(HttpStatus::is4xxClientError) { ... }
.onStatus(HttpStatus::is5xxServerError) { ... }
.awaitBody<Person>()
方法 exchangeToMono()
和 exchangeToFlux()
用来处理需要更多控制的用例,例如根据不同的响应状态来进行解码响应体:
Java.
Mono<Object> entityMono = client.get()
.uri("/persons/1")
.accept(MediaType.APPLICATION_JSON)
.exchangeToMono(response -> {
if (response.statusCode().equals(HttpStatus.OK)) {
return response.bodyToMono(Person.class);
}
else if (response.statusCode().is4xxClientError()) {
// Suppress error status code
return response.bodyToMono(ErrorContainer.class);
}
else {
// Turn to error
return response.createException().flatMap(Mono::error);
}
});
Kotlin.
val entity = client.get()
.uri("/persons/1")
.accept(MediaType.APPLICATION_JSON)
.awaitExchange {
if (response.statusCode() == HttpStatus.OK) {
return response.awaitBody<Person>()
}
else if (response.statusCode().is4xxClientError) {
return response.awaitBody<ErrorContainer>()
}
else {
throw response.createExceptionAndAwait()
}
}
当按照上述方式进行响应处理时,在返回Mono
或者 Flux
完成后,响应体会被检查,如果没有被消耗掉会自动进行释放以防止内存和连接泄露。因此响应不能再被下游进行解码处理。如果需要的话,必须由提供的函数进行响应体的解码操作。
请求体可以来自任何可以被 ReactiveAdapterRegistry
处理的异步类型,例如 Mono
或者Kotlin协程 Deferred
类型,如下示例:
Java.
Mono<Person> personMono = ... ;
Mono<Void> result = client.post()
.uri("/persons/{id}", id)
.contentType(MediaType.APPLICATION_JSON)
.body(personMono, Person.class)
.retrieve()
.bodyToMono(Void.class);
Kotlin.
val personDeferred: Deferred<Person> = ...
client.post()
.uri("/persons/{id}", id)
.contentType(MediaType.APPLICATION_JSON)
.body<Person>(personDeferred)
.retrieve()
.awaitBody<Unit>()
你也可以使用一个对象流,如下所示:
Java.
Flux<Person> personFlux = ... ;
Mono<Void> result = client.post()
.uri("/persons/{id}", id)
.contentType(MediaType.APPLICATION_STREAM_JSON)
.body(personFlux, Person.class)
.retrieve()
.bodyToMono(Void.class);
Kotlin.
val people: Flow<Person> = ...
client.post()
.uri("/persons/{id}", id)
.contentType(MediaType.APPLICATION_JSON)
.body(people)
.retrieve()
.awaitBody<Unit>()
如果你也对象实际的值,也可以直接使用 bodyValue
方法,如下所示:
Java.
Person person = ... ;
Mono<Void> result = client.post()
.uri("/persons/{id}", id)
.contentType(MediaType.APPLICATION_JSON)
.bodyValue(person)
.retrieve()
.bodyToMono(Void.class);
Kotlin.
val person: Person = ...
client.post()
.uri("/persons/{id}", id)
.contentType(MediaType.APPLICATION_JSON)
.bodyValue(person)
.retrieve()
.awaitBody<Unit>()
为了发送表单数据,你可以提供一个 MultiValueMap<String, String>
对象作为请求体。注意内容类型会被 FormHttpMessageWriter
自动设置为 application/x-www-form-urlencoded
。以下示例展示了如何使用 MultiValueMap<String, String>
:
Java.
MultiValueMap<String, String> formData = ... ;
Mono<Void> result = client.post()
.uri("/path", id)
.bodyValue(formData)
.retrieve()
.bodyToMono(Void.class);
Kotlin.
val formData: MultiValueMap<String, String> = ...
client.post()
.uri("/path", id)
.bodyValue(formData)
.retrieve()
.awaitBody<Unit>()
你也可以使用 BodyInserters
直接提供键值对,如下示例:
Java.
import static org.springframework.web.reactive.function.BodyInserters.*;
Mono<Void> result = client.post()
.uri("/path", id)
.body(fromFormData("k1", "v1").with("k2", "v2"))
.retrieve()
.bodyToMono(Void.class);
Kotlin.
import org.springframework.web.reactive.function.BodyInserters.*
client.post()
.uri("/path", id)
.body(fromFormData("k1", "v1").with("k2", "v2"))
.retrieve()
.awaitBody<Unit>()
要发送multipart数据,你需要提供一个 MultiValueMap<String, ?>
,其值是表示part内容的 Object
实例或表示part内容和首部的 HttpEntity
实例。 MultipartBodyBuilder
提供了一个方便的 API 来准备multipart请求。 下面的例子展示了如何创建一个 MultiValueMap<String, ?>
:
Java.
MultipartBodyBuilder builder = new MultipartBodyBuilder();
builder.part("fieldPart", "fieldValue");
builder.part("filePart1", new FileSystemResource("...logo.png"));
builder.part("jsonPart", new Person("Jason"));
builder.part("myPart", part); // Part from a server request
MultiValueMap<String, HttpEntity<?>> parts = builder.build();
Kotlin.
val builder = MultipartBodyBuilder().apply {
part("fieldPart", "fieldValue")
part("filePart1", new FileSystemResource("...logo.png"))
part("jsonPart", new Person("Jason"))
part("myPart", part) // Part from a server request
}
val parts = builder.build()
在大多数情况下,您不必为每个part指定 Content-Type
。 内容类型是根据为序列化选择的 HttpMessageWriter
自动确定的,或者在 Resource
的情况下,根据文件扩展名自动确定。 如有必要,可以显示的通过重载的构建器 part
方法提供用于每个part的 MediaType
。
准备好MultiValueMap
后,将其传递给 WebClient
的最简单方法是通过 body
方法,如下例所示:
Java.
MultipartBodyBuilder builder = ...;
Mono<Void> result = client.post()
.uri("/path", id)
.body(builder.build())
.retrieve()
.bodyToMono(Void.class);
Kotlin.
val builder: MultipartBodyBuilder = ...
client.post()
.uri("/path", id)
.body(builder.build())
.retrieve()
.awaitBody<Unit>()
如果 MultiValueMap
包含至少一个非 String
值,该值也可以表示常规表单数据(即 application/x-www-form-urlencoded
),则不需要设置 Content-Type
为 multipart/form-data
。 使用 MultipartBodyBuilder
时总是如此,它确保使用了一个 HttpEntity
包装器。
作为 MultipartBodyBuilder
的替代方案,您还可以通过内置的 BodyInserters
提供内联样式的multipart内容,如下例所示:
Java.
import static org.springframework.web.reactive.function.BodyInserters.*;
Mono<Void> result = client.post()
.uri("/path", id)
.body(fromMultipartData("fieldPart", "value").with("filePart", resource))
.retrieve()
.bodyToMono(Void.class);
Kotlin.
import org.springframework.web.reactive.function.BodyInserters.*
client.post()
.uri("/path", id)
.body(fromMultipartData("fieldPart", "value").with("filePart", resource))
.retrieve()
.awaitBody<Unit>()
可以通过 WebClient.Builder
注册一个客户端过滤器 (ExchangeFilterFunction
) 来拦截和修改请求,如下例所示:
Java.
WebClient client = WebClient.builder()
.filter((request, next) -> {
ClientRequest filtered = ClientRequest.from(request)
.header("foo", "bar")
.build();
return next.exchange(filtered);
})
.build();
Kotlin.
val client = WebClient.builder()
.filter { request, next ->
val filtered = ClientRequest.from(request)
.header("foo", "bar")
.build()
next.exchange(filtered)
}
.build()
这可用于横切关注点,例如鉴权。 以下示例通过静态工厂方法使用过滤器进行基本身份验证:
Java.
import static org.springframework.web.reactive.function.client.ExchangeFilterFunctions.basicAuthentication;
WebClient client = WebClient.builder()
.filter(basicAuthentication("user", "password"))
.build();
Kotlin.
import org.springframework.web.reactive.function.client.ExchangeFilterFunctions.basicAuthentication
val client = WebClient.builder()
.filter(basicAuthentication("user", "password"))
.build()
可以通过改变现有的 WebClient
实例来添加或删除过滤器,从而产生一个不影响原始实例的新 WebClient
实例。 例如:
Java.
import static org.springframework.web.reactive.function.client.ExchangeFilterFunctions.basicAuthentication;
WebClient client = webClient.mutate()
.filters(filterList -> {
filterList.add(0, basicAuthentication("user", "password"));
})
.build();
Kotlin.
val client = webClient.mutate()
.filters { it.add(0, basicAuthentication("user", "password")) }
.build()
WebClient
是一个围绕过滤器链的薄门面,后跟一个 ExchangeFunction
。 它提供了一个工作流来发出请求、编解码更高抽象级别的对象,并确保响应内容被消费掉。 当过滤器以某种方式处理响应时,必须格外小心始终确保其内容被消费掉或者将其向下游传递到 WebClient
。 下面是一个处理 UNAUTHORIZED
状态代码的过滤器,但确保任何响应内容,无论是否是预期的,都被释放:
Java.
public ExchangeFilterFunction renewTokenFilter() {
return (request, next) -> next.exchange(request).flatMap(response -> {
if (response.statusCode().value() == HttpStatus.UNAUTHORIZED.value()) {
return response.releaseBody()
.then(renewToken())
.flatMap(token -> {
ClientRequest newRequest = ClientRequest.from(request).build();
return next.exchange(newRequest);
});
} else {
return Mono.just(response);
}
});
}
Kotlin.
fun renewTokenFilter(): ExchangeFilterFunction? {
return ExchangeFilterFunction { request: ClientRequest?, next: ExchangeFunction ->
next.exchange(request!!).flatMap { response: ClientResponse ->
if (response.statusCode().value() == HttpStatus.UNAUTHORIZED.value()) {
return@flatMap response.releaseBody()
.then(renewToken())
.flatMap { token: String? ->
val newRequest = ClientRequest.from(request).build()
next.exchange(newRequest)
}
} else {
return@flatMap Mono.just(response)
}
}
}
}
可以向请求添加属性。 如果你想通过过滤器链传递信息并影响给定请求的过滤器行为,这很方便。 例如:
Java.
WebClient client = WebClient.builder()
.filter((request, next) -> {
Optional<Object> usr = request.attribute("myAttribute");
// ...
})
.build();
client.get().uri("https://example.org/")
.attribute("myAttribute", "...")
.retrieve()
.bodyToMono(Void.class);
}
Kotlin.
val client = WebClient.builder()
.filter { request, _ ->
val usr = request.attributes()["myAttribute"];
// ...
}
.build()
client.get().uri("https://example.org/")
.attribute("myAttribute", "...")
.retrieve()
.awaitBody<Unit>()
请注意,你可以在 WebClient.Builder
级别全局配置 defaultRequest
回调,它允许您将属性插入到所有请求中,例如可以在 Spring MVC 应用程序中使用以根据 ThreadLocal
数据填充请求属性。
Attributes 提供了一种向过滤器链传递信息的便捷方式,但它们只影响当前请求。 如果您想传递传播到其他嵌套请求的信息,例如 通过flatMap
,或之后执行,例如 通过concatMap
,那么你需要使用ReactorContext
。
Reactor Context
需要填充在反应链的末尾,以便应用于所有操作。 例如:
Java.
WebClient client = WebClient.builder()
.filter((request, next) ->
Mono.deferContextual(contextView -> {
String value = contextView.get("foo");
// ...
}))
.build();
client.get().uri("https://example.org/")
.retrieve()
.bodyToMono(String.class)
.flatMap(body -> {
// perform nested request (context propagates automatically)...
})
.contextWrite(context -> context.put("foo", ...));
WebClient
可以通过在结尾处阻塞以便在同步风格的代码中使用:
Java.
Person person = client.get().uri("/person/{id}", i).retrieve()
.bodyToMono(Person.class)
.block();
List<Person> persons = client.get().uri("/persons").retrieve()
.bodyToFlux(Person.class)
.collectList()
.block();
Kotlin.
val person = runBlocking {
client.get().uri("/person/{id}", i).retrieve()
.awaitBody<Person>()
}
val persons = runBlocking {
client.get().uri("/persons").retrieve()
.bodyToFlow<Person>()
.toList()
}
但是,如果需要进行多次调用,避免单独阻塞每个响应,而是等待组合结果更有效:
Java.
Mono<Person> personMono = client.get().uri("/person/{id}", personId)
.retrieve().bodyToMono(Person.class);
Mono<List<Hobby>> hobbiesMono = client.get().uri("/person/{id}/hobbies", personId)
.retrieve().bodyToFlux(Hobby.class).collectList();
Map<String, Object> data = Mono.zip(personMono, hobbiesMono, (person, hobbies) -> {
Map<String, String> map = new LinkedHashMap<>();
map.put("person", person);
map.put("hobbies", hobbies);
return map;
})
.block();
Kotlin.
val data = runBlocking {
val personDeferred = async {
client.get().uri("/person/{id}", personId)
.retrieve().awaitBody<Person>()
}
val hobbiesDeferred = async {
client.get().uri("/person/{id}/hobbies", personId)
.retrieve().bodyToFlow<Hobby>().toList()
}
mapOf("person" to personDeferred.await(), "hobbies" to hobbiesDeferred.await())
}
以上只是一个例子。 有许多其他模式和操作符可以组合一个反应式管道,这些管道可以进行许多远程调用,可能是一些嵌套的、相互依赖的,直到最后都不会阻塞。
注意
使用
Flux
或Mono
,你永远不必在 Spring MVC 或 Spring WebFlux 控制器中阻塞。 简单地从控制器方法返回结果反应类型。 相同的原则适用于 Kotlin Coroutines 和 Spring WebFlux,只需在控制器方法中使用挂起函数或返回Flow
。
要测试使用 WebClient
的代码,您可以使用模拟 Web 服务器,例如 OkHttp MockWebServer。 要查看其使用示例,请查看 Spring Framework 测试套件中的 WebClientIntegrationTests
或 OkHttp 存储库中的 static-server
示例。