一、前言

  WebClient是Spring 5中引入的一个新的非阻塞、响应式HTTP客户端,用于发送HTTP请求和接收响应。它基于Reactor和Netty,支持异步调用,并提供了简洁且易于使用的API。WebClient是Spring WebFlux模块的一部分,从Spring 5.0开始,它作为RestTemplate的替代品,具有更好的响应式能力和异步支持。

二、WebClient的主要特点

  非阻塞:WebClient采用响应式编程模型,支持非阻塞IO操作,使得在高并发场景下性能更佳。

  异步调用:支持异步发送HTTP请求,并通过回调函数处理响应结果,使得在等待响应时不会阻塞当前线程。

  简洁易用:提供了简洁的API用于构建和发送HTTP请求,使得编写HTTP客户端代码更加容易。

  灵活性:允许自定义请求的各个方面,包括请求头、请求体、URI变量等。

三、WebClient的常见方法

  post():创建一个POST请求。

  put():创建一个PUT请求。

  get():创建一个GET请求。

  delete():创建一个DELETE请求。

  uri(String uri):设置请求的URI地址。

  header(String headerName, String… headerValues):设置请求头。

  headers(Consumer headersConsumer):使用Consumer对象来自定义请求头。

  accept(MediaType… acceptableMediaTypes):设置期望的响应内容类型。

  contentType(MediaType contentType):设置请求的内容类型。

  body(BodyInserter<?, ? super ClientHttpRequest> bodyInserter):设置请求的内容体,使用BodyInserter对象。

  retrieve():发送请求并获取响应。

四、WebClient方法代码示例

  使用WebClient进行HTTP请求时,以下是GET请求(包括带参数的GET请求)和POST请求(发送JSON或表单数据)的代码示例。

4.1 GET 请求(不带参数的GET请求)

不带参数的GET请求

import org.springframework.web.reactive.function.client.WebClient;  
  
public class WebClientExample {  
  
    public static void main(String[] args) {  
        WebClient webClient = WebClient.create("http://example.com");  
  
        webClient.get()  
                .uri("/api/resource")  
                .retrieve()  
                .bodyToMono(String.class)  
                .subscribe(  
                        result -> System.out.println("Result: " + result),  
                        error -> System.err.println("Error: " + error)  
                );  
    }  
}

4.2 GET请求(带参数的使用URI模板变量)

import org.springframework.web.reactive.function.client.WebClient;  
  
public class WebClientExample {  
  
    public static void main(String[] args) {  
        WebClient webClient = WebClient.create("http://example.com");  
  
        String id = "123";  
  
        webClient.get()  
                .uri("/api/resource/{id}", id) // 使用URI模板变量  
                .retrieve()  
                .bodyToMono(String.class)  
                .subscribe(  
                        result -> System.out.println("Result: " + result),  
                        error -> System.err.println("Error: " + error)  
                );  
    }  
}

4.3 GET请求(带参数的使用URI查询参数)

import org.springframework.web.reactive.function.client.WebClient;  
  
public class WebClientExample {  
  
    public static void main(String[] args) {  
        WebClient webClient = WebClient.create("http://example.com");  
  
        String id = "123";  
  
        webClient.get()  
                .uri(uriBuilder -> uriBuilder  
                        .path("/api/resource")  
                        .queryParam("id", id) // 添加查询参数  
                        .build())  
                .retrieve()  
                .bodyToMono(String.class)  
                .subscribe(  
                        result -> System.out.println("Result: " + result),  
                        error -> System.err.println("Error: " + error)  
                );  
    }  
}

4.4 POST 请求(发送JSON数据)

import org.springframework.http.MediaType;  
import org.springframework.web.reactive.function.client.WebClient;  
  
public class WebClientExample {  
  
    public static void main(String[] args) {  
        WebClient webClient = WebClient.create("http://example.com");  
  
        String jsonData = "{\"name\":\"John\", \"age\":30}";  
  
        webClient.post()  
                .uri("/api/resource")  
                .contentType(MediaType.APPLICATION_JSON)  
                .bodyValue(jsonData)  
                .retrieve()  
                .bodyToMono(String.class)  
                .subscribe(  
                        result -> System.out.println("Result: " + result),  
                        error -> System.err.println("Error: " + error)  
                );  
    }  
}

4.5 POST 请求(发送表单数据,使用MultiValueMap)

import org.springframework.http.MediaType;  
import org.springframework.util.LinkedMultiValueMap;  
import org.springframework.util.MultiValueMap;  
import org.springframework.web.reactive.function.client.WebClient;  
  
public class WebClientExample {  
  
    public static void main(String[] args) {  
        WebClient webClient = WebClient.create("http://example.com");  
  
        MultiValueMap<String, String> formData = new LinkedMultiValueMap<>();  
        formData.add("name", "John");  
        formData.add("age", "30");  
  
        webClient.post()  
                .uri("/api/resource")  
                .contentType(MediaType.APPLICATION_FORM_URLENCODED)  
                .bodyValue(formData)  
                .retrieve()  
                .bodyToMono(String.class)  
                .subscribe(  
                        result -> System.out.println("Result: " + result),  
                        error -> System.err.println("Error: " + error)  
                );  
    }  
}
请注意,以上代码示例中的subscribe方法用于触发请求

四、WebClient异步与同步

  当使用WebClient时,通常推荐的做法是使用其异步API,因为它基于响应式编程模型,并且可以充分利用非阻塞IO来提高性能和吞吐量。但是,有时为了简化代码或满足特定需求,你可能想要以同步的方式使用WebClient。然而,请注意,将响应式编程转换为同步调用可能会牺牲性能和响应性。

异步代码示例
  异步代码示例使用Mono或Flux类型来处理响应,并使用block()方法(尽管在生产代码中不推荐,因为它会阻塞当前线程)来等待结果。但通常,你会使用.subscribe()方法来异步处理响应。

import org.springframework.http.MediaType;  
import org.springframework.web.reactive.function.client.WebClient;  
import reactor.core.publisher.Mono;  
  
public class WebClientAsyncExample {  
  
    public static void main(String[] args) {  
        WebClient webClient = WebClient.create("http://example.com");  
  
        Mono<String> responseMono = webClient.get()  
                .uri("/api/resource")  
                .accept(MediaType.APPLICATION_JSON)  
                .retrieve()  
                .bodyToMono(String.class);  
  
        // 使用subscribe异步处理响应  
        responseMono.subscribe(  
            result -> System.out.println("Result: " + result),  
            error -> System.err.println("Error: " + error)  
        );  
  
        // 注意:通常不推荐在main线程中调用block(),因为它会阻塞  
        // 以下代码仅用于演示目的  
        String result = responseMono.block(); // 阻塞当前线程直到结果可用  
        System.out.println("Blocked Result: " + result);  
    }  
}

同步代码示例
  由于WebClient本身并不直接支持同步调用,但你可以通过封装异步调用来模拟同步行为。这通常涉及到使用block()方法,但如前所述,这不应该在生产代码中使用,因为它会阻塞当前线程。

import org.springframework.http.MediaType;  
import org.springframework.web.reactive.function.client.WebClient;  
import reactor.core.publisher.Mono;  
  
public class WebClientSyncExample {  
  
    public static void main(String[] args) {  
        WebClient webClient = WebClient.create("http://example.com");  
  
        // 使用block()方法阻塞当前线程以等待结果  
        String result = webClient.get()  
                .uri("/api/resource")  
                .accept(MediaType.APPLICATION_JSON)  
                .retrieve()  
                .bodyToMono(String.class)  
                .block(); // 注意:这会阻塞当前线程  
  
        System.out.println("Result: " + result);  
    }  
}

  在这个同步代码示例中,我们使用block()方法来等待异步操作完成并返回结果。但是,这会导致当前线程被阻塞,直到结果可用为止。这可能会降低应用程序的性能和响应性,因此应谨慎使用。在生产环境中,通常建议使用异步编程模型来处理HTTP请求。

  以上是Spring 5.0 中引入的一个新的非阻塞、响应式HTTP客户端WebClient,用于发送HTTP请求和接收响应的一些介绍与示例。

Logo

开放原子开发者工作坊旨在鼓励更多人参与开源活动,与志同道合的开发者们相互交流开发经验、分享开发心得、获取前沿技术趋势。工作坊有多种形式的开发者活动,如meetup、训练营等,主打技术交流,干货满满,真诚地邀请各位开发者共同参与!

更多推荐