自 Spring 5 以來,WebClient已成為Spring WebFlux的一部分,并且是發(fā)出 HTTP 請求的首選方式。它是經(jīng)典RestTemplate的首選替代方案,后者自 Spring 5.0 以來一直處于維護模式。
本文將討論 Spring WebClient和RestTemplate類之間的主要區(qū)別。
特征 | WebClient | RestTemplate |
反應(yīng)式編程 | 基于反應(yīng)式原則構(gòu)建并支持反應(yīng)式編程。 | 同步而不是為反應(yīng)式編程而設(shè)計。 |
技術(shù) | 構(gòu)建在反應(yīng)式技術(shù)棧上。 | 構(gòu)建在 Servlet 技術(shù)棧上。 |
線程模型 | 采用非阻塞I/O,適合處理大量并發(fā)請求。 | 使用阻塞 I/O,在高并發(fā)場景下可能會導(dǎo)致線程阻塞。 |
Java版本 | 需要 Java 8+ 或更高版本。支持函數(shù)式編程。 | 與 Java 6+ 或更高版本兼容。 |
錯誤處理 | 使用onErrorResume、onErrorReturn等運算符提供強大的錯誤處理。 | 錯誤處理通常使用 try-catch 塊完成。 |
流媒體 | 支持使用Flux和Mono流式傳輸數(shù)據(jù),適合反應(yīng)式流式場景。 | 對流的支持有限,不太適合反應(yīng)式流。 |
用例 | 最適合微服務(wù)、反應(yīng)式應(yīng)用以及需要高并發(fā)的場景。 | 適用于傳統(tǒng)的整體應(yīng)用程序和簡單的用例。 |
依賴關(guān)系 | 需要Spring WebFlux依賴項。 | 需要Spring Web依賴。 |
未來的支持 | 與反應(yīng)式編程模型保持一致,并可能會得到持續(xù)的開發(fā)和支持。 | 可能會維護更新,將來可能不會受到那么多關(guān)注。 |
RestTemplate本質(zhì)上是阻塞的,并使用 Java Servlet API 的每個請求一個線程模型。這意味著RestTemplate一旦向遠程服務(wù)器發(fā)送請求,就會等待響應(yīng)。默認情況下,每次RestTemplate都會創(chuàng)建新的,并在收到并處理響應(yīng)后關(guān)閉連接。Httpconnection 創(chuàng)建和關(guān)閉 URL 連接是一項成本高昂的操作。為了在生產(chǎn)類應(yīng)用程序中有效地使用RestTemplate ,我們必須使用HTTP 連接池,否則性能會快速下降。當應(yīng)用程序中有大量請求時,線程和連接的數(shù)量也會按比例增加。這會給服務(wù)器資源帶來負擔。如果服務(wù)器速度緩慢,用戶很快就會發(fā)現(xiàn)應(yīng)用程序性能下降,甚至無響應(yīng)。
請注意,RestTemplate 是線程安全的,并且可以隨時在多個連接之間共享單個實例。
@Servicepublic class MyService { private final RestTemplate restTemplate; @Autowired public MyService(RestTemplate restTemplate) { this.restTemplate = restTemplate; } public String getData() { ResponseEntity<String> responseEntity = restTemplate.getForEntity("https://api.example.com/data", String.class); String responseBody = responseEntity.getBody(); return responseBody; }}
與RestTemplate相反,WebClient本質(zhì)上是異步且非阻塞的。它遵循 Spring WebFlux 反應(yīng)式框架的事件驅(qū)動架構(gòu)。使用WebClient,客戶端無需等待響應(yīng)返回。相反,當服務(wù)器有響應(yīng)時,它將使用回調(diào)方法收到通知。
當我們通過WebClient調(diào)用返回 Mono或 Flux 的API 時,API 會立即返回。而調(diào)用結(jié)果將通過 mono 或 flux 回調(diào)傳遞給調(diào)用端。
請注意,如果需要,我們可以通過WebClient.block()方法實現(xiàn)類似RestTemplate的同步處理。
@Servicepublic class MyService { private final WebClient webClient; @Autowired public MyService(WebClient webClient) { this.webClient = webClient; } public Mono<String> getData() { return webClient.get() .uri("/data") .retrieve() .bodyToMono(String.class) .subscribe( // onSuccess callback result -> { System.out.println("Success: " + result); }, // onError callback error -> { System.err.println("Error: " + error.getMessage()); } ); }}
從上面可以清楚地看出, WebClient和RestTemplate之間唯一的大區(qū)別是它們的阻塞性質(zhì)。RestTemplate會阻止請求線程,而WebClient不會。我們可以使用WebClient來發(fā)出同步請求,但反之則不行。RestTemplate無法發(fā)出異步請求。
雖然WebClient是未來使用的首選方式,但 RestTemplate 應(yīng)該會長期保留,盡管沒有添加任何新的核心功能。
在考慮使用WebClient 構(gòu)建新應(yīng)用程序時,我們必須記住,要構(gòu)建真正的非阻塞應(yīng)用程序,必須以非阻塞方式創(chuàng)建/使用其所有組件,即客戶端、控制器、中間服務(wù),甚至數(shù)據(jù)庫。如果其中之一阻塞了請求,目的就會落空。
本文鏈接:http://www.tebozhan.com/showinfo-26-14531-0.html一文讀懂WebClient和RestTemplate的差異
聲明:本網(wǎng)頁內(nèi)容旨在傳播知識,若有侵權(quán)等問題請及時與本網(wǎng)聯(lián)系,我們將在第一時間刪除處理。郵件:2376512515@qq.com