国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

強大又優雅!Spring Boot 中 RestTemplate 的最佳實踐詳解

開發 前端
我們將詳細演示如何使用 RestTemplate? 進行 POST 請求,包括如何設置請求頭和請求體、如何構建和傳遞復雜數據,以及如何處理返回的響應。

在現代開發中,API 的設計和調用變得尤為重要,尤其是基于 REST 架構的服務調用。RestTemplate 是 Spring 提供的用于同步調用 RESTful 服務的強大工具,它支持各種 HTTP 方法,如 GET、POST、PUT、DELETE 等。作為開發者,理解并掌握如何高效使用 RestTemplate 是優化服務交互性能的重要一步。本文旨在深入探討 RestTemplate 的 POST 請求方法以及 exchange() 和 execute() 等常用方法,幫助你在實際開發中靈活使用這些工具應對復雜的業務場景。

通過示例代碼,我們將詳細演示如何使用 RestTemplate 進行 POST 請求,包括如何設置請求頭和請求體、如何構建和傳遞復雜數據,以及如何處理返回的響應。同時,我們還將探索如何使用 exchange() 指定 HTTP 方法,實現靈活的請求處理。無論是初學者還是有經驗的開發者,這篇文章都將為你提供有價值的參考。

RestTemplate 是 Spring 提供的用于訪問 REST 服務的客戶端。

它提供了多種便捷的方法來訪問遠程 HTTP 服務,大大提高了客戶端代碼開發的效率。

之前,我使用 Apache 的 HttpClient 開發 HTTP 請求。代碼非常復雜,我不得不管理資源清理等問題,代碼繁瑣且包含大量冗余部分。

以下是我封裝的一個 post 請求工具的截圖:

圖片圖片

本教程將指導你如何在 Spring 生態系統中使用 RestTemplate 進行 GET 和 POST 請求,并通過 exchange 方法來指定請求類型。同時,還會分析 RestTemplate 的核心方法。

閱讀完本教程后,你將能夠以優雅的方式進行 HTTP 請求。

RestTemplate 簡介

*RestTemplate* 是 *Spring* 中用于同步客戶端通信的核心類。它簡化了與 HTTP 服務的通信,并遵循 RestFul 原則。代碼只需提供一個 URL 并提取結果。

默認情況下,RestTemplate 依賴于 JDK 的 HTTP 連接工具。但是,你可以通過 setRequestFactory 屬性切換到其他 HTTP 工具源,例如 Apache HttpComponents、Netty 和 OkHttp。

RestTemplate 大大簡化了表單數據的提交,并包含對 JSON 數據的自動轉換。

然而,要真正掌握它的使用,必須理解 HttpEntity 的結構(包括 headers 和 body)以及它與 uriVariables 之間的區別。

這一點在 POST 請求中尤為明顯,稍后我們將詳細討論。

此類的主要入口點基于六種 HTTP 方法:

圖片圖片

此外,exchange 和 execute 可以與上述方法互換使用。

在內部,RestTemplate 默認使用 HttpMessageConverter 實例,將 HTTP 消息轉換為 POJO或將 POJO 轉換為 HTTP 消息。默認情況下,它會為主要的 MIME 類型注冊轉換器,但你也可以通過 setMessageConverters 注冊其他轉換器。

(在實際使用中,這一點并不十分明顯;許多方法都有一個 responseType 參數,你可以傳遞一個與響應體映射的對象,底層的 HttpMessageConverter 會進行映射。)

HttpMessageConverterExtractor<T> responseExtractor =
                new HttpMessageConverterExtractor<>(responseType, getMessageConverters(), logger);

HttpMessageConverter.java 源代碼:

public interface HttpMessageConverter<T> {
        // 判斷該轉換器是否可以讀取給定的類。
    boolean canRead(Class<?> clazz, @Nullable MediaType mediaType);

        // 判斷該轉換器是否可以寫入給定的類。
    boolean canWrite(Class<?> clazz, @Nullable MediaType mediaType);

        // 返回一個 List<MediaType>
    List<MediaType> getSupportedMediaTypes();

        // 讀取輸入消息
    T read(Class<? extends T> clazz, HttpInputMessage inputMessage)
            throws IOException, HttpMessageNotReadableException;

        // 將對象寫入輸出消息
    void write(T t, @Nullable MediaType contentType, HttpOutputMessage outputMessage)
            throws IOException, HttpMessageNotWritableException;
}

在內部,RestTemplate 默認使用 SimpleClientHttpRequestFactory 和 DefaultResponseErrorHandler 來分別處理 HTTP 請求的創建和錯誤處理。

然而,你可以通過 setRequestFactory 和 setErrorHandler 來覆蓋這些默認行為。

GET 請求

getForObject() 方法

public <T> T getForObject(String url, Class<T> responseType, Object... uriVariables){}
public <T> T getForObject(String url, Class<T> responseType, Map<String, ?> uriVariables)
public <T> T getForObject(URI url, Class<T> responseType)

getForObject() 方法可以將 HTTP 響應直接轉換為 POJO,而不像 getForEntity() 需要額外處理響應。

getForObject 會直接返回 POJO,省略了大量響應信息。

POJO 示例:

public class Notice {
    private int status;
    private Object msg;
    private List<DataBean> data;
}
public class DataBean {
  private int noticeId;
  private String noticeTitle;
  private Object noticeImg;
  private long noticeCreateTime;
  private long noticeUpdateTime;
  private String noticeContent;
}

無參數的 GET 請求

/**
 * 無參數的 GET 請求
 */
@Test
public void restTemplateGetTest(){
    RestTemplate restTemplate = new RestTemplate();
    Notice notice = restTemplate.getForObject("http://icoderoad.com/notice/list/1/5", Notice.class);
    System.out.println(notice);
}

控制臺輸出:

INFO 19076 --- [           main] c.w.s.c.w.c.HelloControllerTest          
: Started HelloControllerTest in 5.532 seconds (JVM running for 7.233)

Notice{status=200, msg=null, data=[DataBean{noticeId=21, noticeTitle='aaa', noticeImg=null, 
noticeCreateTime=1525292723000, noticeUpdateTime=1525292723000, noticeContent='<p>aaa</p>'}, 
DataBean{noticeId=20, noticeTitle='ahaha', noticeImg=null, noticeCreateTime=1525291492000, 
noticeUpdateTime=1525291492000, noticeContent='<p>ah.......'

帶參數的 GET 請求 1

Notice notice = restTemplate.getForObject("http://icoderoad.com/notice/list/{1}/{2}", Notice.class, 1, 5);

可以看到,使用了占位符 {1} 和 {2}。

帶參數的 GET 請求 2

Map<String, String> map = new HashMap<>();
map.put("start", "1");
map.put("page", "5");
Notice notice = restTemplate.getForObject("http://icoderoad.com/notice/list/", Notice.class, map);

在這種情況下,使用 Map 加載參數,默認情況下會以 PathVariable 格式解析 URL。

getForEntity() 方法

public <T> ResponseEntity<T> getForEntity(String url, Class<T> responseType, Object... uriVariables){}
public <T> ResponseEntity<T> getForEntity(String url, Class<T> responseType, Map<String, ?> uriVariables){}
public <T> ResponseEntity<T> getForEntity(URI url, Class<T> responseType){}

與 getForObject() 不同,此方法返回 ResponseEntity 對象。

如果需要將其轉換為 POJO,則必須使用 JSON 工具類,可根據個人偏好選擇。

對于不熟悉解析 JSON 的人,可以查閱工具如 FastJson 或 Jackson。接下來我們來探討 ResponseEntity 的相關方法。

ResponseEntity、HttpStatus 和 BodyBuilder 的結構

//ResponseEntity.java

public HttpStatus getStatusCode(){}
public int getStatusCodeValue(){}
public boolean equals(@Nullable Object other) {}
public String toString() {}
public static BodyBuilder status(HttpStatus status) {}
public static BodyBuilder ok() {}
public static <T> ResponseEntity<T> ok(T body) {}
public static BodyBuilder created(URI location) {}
...
//HttpStatus.java

public enum HttpStatus {
public boolean is1xxInformational() {}
public boolean is2xxSuccessful() {}
public boolean is3xxRedirection() {}
public boolean is4xxClientError() {}
public boolean is5xxServerError() {}
public boolean isError() {}
}
//BodyBuilder.java

public interface BodyBuilder extends HeadersBuilder<BodyBuilder> {
    // 通過 Content-Length 頭設置響應實體的內容長度
    BodyBuilder contentLength(long contentLength);
    // 設置響應實體的 MediaType
    BodyBuilder contentType(MediaType contentType);
    // 設置響應實體的內容并返回
    <T> ResponseEntity<T> body(@Nullable T body);
}

如上所示,ResponseEntity 包含了來自 HttpStatus 和 BodyBuilder 的信息,這使得處理原始響應變得更加簡單。

示例:

@Test
public void rtGetEntity(){
    RestTemplate restTemplate = new RestTemplate();
    ResponseEntity<Notice> entity = restTemplate.getForEntity("http://icoderoad.com/notice/list/1/5", Notice.class);

    HttpStatus statusCode = entity.getStatusCode();
    System.out.println("statusCode.is2xxSuccessful()" + statusCode.is2xxSuccessful());

    Notice body = entity.getBody();
    System.out.println("entity.getBody()" + body);

    ResponseEntity.BodyBuilder status = ResponseEntity.status(statusCode);
    status.contentLength(100);
    status.body("在此處添加一個聲明");
    ResponseEntity<Class<Notice>> body1 = status.body(Notice.class);
    Class<Notice> body2 = body1.getBody();
    System.out.println("body1.toString()" + body1.toString());
}

輸出:

statusCode.is2xxSuccessful() true
entity.getBody() Notice{status=200, msg=null, data=[DataBean{noticeId=21, noticeTitle='aaa', ...
body1.toString() <200 OK, class com.waylau.spring.cloud.weather.pojo.Notice, {Content-Length=[100]}>

當然,像 getHeaders() 這樣的方法也是可用的,但在此不作示例。

POST 請求

類似于 GET 請求,POST 請求也有 postForObject 和 postForEntity 方法。

public <T> T postForObject(String url, @Nullable Object request, Class<T> responseType, Object... uriVariables)
         throws RestClientException {}
public <T> T postForObject(String url, @Nullable Object request, Class<T> responseType, Map<String, ?> uriVariables)
         throws RestClientException {}
public <T> T postForObject(URI url, @Nullable Object request, Class<T> responseType) throws RestClientException {}

示例

這里我使用一個郵箱驗證接口進行測試。

@Test
public void rtPostObject(){
    RestTemplate restTemplate = new RestTemplate();
    String url = "http://47.xxx.xxx.96/register/checkEmail";
    HttpHeaders headers = new HttpHeaders();
    headers.setContentType(MediaType.APPLICATION_FORM_URLENCODED);
    MultiValueMap<String, String> map = new LinkedMultiValueMap<>();
    map.add("email", "844072586@qq.com");

    HttpEntity<MultiValueMap<String, String>> request = new HttpEntity<>(map, headers);
    ResponseEntity<String> response = restTemplate.postForEntity(url, request, String.class);
    System.out.println(response.getBody());
}

執行結果:

{"status":500,"msg":"該郵箱已經注冊","data":null}

為什么使用 MultiValueMap?

MultiValueMap 是 Map 的子類,它允許每個鍵存儲多個值。這里簡單介紹該接口:

public interface MultiValueMap<K, V> extends Map<K, List<V>> {...}

使用 MultiValueMap 的原因是 HttpEntity 接受的請求類型為 MultiValueMap:

public HttpEntity(@Nullable T body, @Nullable MultiValueMap<String, String> headers){}

在這個構造函數中,我們傳入的 map 是請求體,headers 是請求頭。

我們使用 HttpEntity 是因為,盡管 restTemplate.postForEntity 方法似乎接受 @Nullable Object request 類型,但如果深入追溯,會發現這個 request 是通過 HttpEntity 解析的。核心代碼如下:

if (requestBody instanceof HttpEntity) {
    this.requestEntity = (HttpEntity<?>) requestBody;
} else if (requestBody != null) {
    this.requestEntity = new HttpEntity<>(requestBody);
} else {
    this.requestEntity = HttpEntity.EMPTY;
}

我曾嘗試使用 map 傳遞參數,雖然編譯時沒有報錯,但請求無效,最終出現 400 錯誤。

這種請求方式已經滿足 POST 請求的需求,同時,cookie 作為請求頭的一部分,也可以根據需要進行設置。

其他方法與此類似。

使用 exchange 指定 HTTP 方法

exchange() 方法與 getForObject()、getForEntity()、postForObject() 和 postForEntity() 不同之處在于它允許你指定 HTTP 方法。

你會注意到,所有的 exchange 方法似乎都有 @Nullable HttpEntity<?> requestEntity 參數,這意味著我們需要使用 HttpEntity 來傳遞請求體。正如前面提到的,使用 HttpEntity性能更好。

示例

@Test
public void rtExchangeTest() throws JSONException {
    RestTemplate restTemplate = new RestTemplate();
    String url = "http://icoderoad.com/notice/list";
    HttpHeaders headers = new HttpHeaders();
    headers.setContentType(MediaType.APPLICATION_FORM_URLENCODED);
    JSONObject jsonObj = new JSONObject();
    jsonObj.put("start", 1);
    jsonObj.put("page", 5);

    HttpEntity<String> entity = new HttpEntity<>(jsonObj.toString(), headers);
    ResponseEntity<JSONObject> exchange = restTemplate.exchange(url, HttpMethod.GET, entity, JSONObject.class);
    System.out.println(exchange.getBody());
}

這次我使用了 JSONObject 來傳遞和返回數據。其他 HttpMethod 方法的使用類似。

使用 execute 指定 HTTP 方法

execute() 方法類似于 exchange(),它允許你指定不同的 HttpMethod 類型。但不同之處在于它返回的響應體是一個對象 <T>,而不是 ResponseEntity<T>。

需要強調的是,execute() 方法是上述所有方法的底層實現。以下是一個示例:

@Override
@Nullable
public <T> T postForObject(String url, @Nullable Object request, Class<T> responseType, Map<String, ?> uriVariables)
        throws RestClientException {

    RequestCallback requestCallback = httpEntityCallback(request, responseType);
    HttpMessageConverterExtractor<T> responseExtractor =
            new HttpMessageConverterExtractor<>(responseType, getMessageConverters(), logger);
    return execute(url, HttpMethod.POST, requestCallback, responseExtractor, uriVariables);
}

結語

通過對 RestTemplate 不同方法的深入講解,特別是 POST 請求的使用以及 exchange()、execute() 的靈活運用,本文展示了在開發過程中如何使用這些方法簡化與外部服務的交互,并提升代碼的可讀性和維護性。在面對復雜的業務需求時,掌握這些技術將幫助開發者在請求數據、處理響應以及提升 API 性能方面取得更好的平衡。

對于高效的 HTTP 請求處理,RestTemplate 提供了豐富的功能,靈活支持多種請求方式和參數配置,極大地簡化了開發流程。隨著項目復雜度的增加,理解和掌握這些工具的使用技巧,能夠大大提升開發效率,同時減少潛在的錯誤。通過深入研究 RestTemplate,我們可以構建出更加健壯、高效的服務交互機制,滿足不斷變化的業務需求。希望本文能為你提供深入的理解,助力你的開發之旅。

責任編輯:武曉燕 來源: 路條編程
相關推薦

2024-10-11 11:46:40

2024-11-21 14:42:31

2024-04-18 08:28:06

2024-05-13 13:13:13

APISpring程序

2024-03-08 10:50:44

Spring技術應用程序

2024-11-28 09:43:04

2020-08-14 10:40:35

RestTemplatRetrofitJava

2025-01-15 08:19:12

SpringBootRedis開源

2024-08-13 08:41:18

2024-09-27 12:27:31

2023-09-22 10:12:57

2024-10-15 10:38:32

2023-09-14 08:16:51

2017-01-15 14:50:34

Spring Batc實踐

2023-12-06 07:13:16

RESTAPI客戶端

2025-07-14 01:30:00

2025-08-26 07:44:57

2024-08-02 09:15:22

Spring捕捉格式

2025-03-11 00:55:00

Spring停機安全

2022-05-25 09:00:00

令牌JWT安全
點贊
收藏

51CTO技術棧公眾號

精品毛片免费观看| 日韩免费一级| 成人免费在线视频播放| 韩国女主播一区二区| 一区二区三区在线看| 狠狠干一区二区| 欧美久久久网站| 久久久久国产精品免费| 日本中文字幕中出在线| 91丝袜美腿高跟国产极品老师 | 欧美中文字幕亚洲一区二区va在线| 韩漫成人漫画| 午夜一区二区三区视频| 日韩在线观看a| 国产精品二区不卡| 欧美激情中文网| 97se综合| 欧美夫妻性生活| **毛片在线网站| 欧美激情一区二区三区在线| 亚洲av综合色区| 日本免费新一区视频| 国产精品a久久久久久| 日韩福利影视| 在线看福利67194| 19禁羞羞电影院在线观看| 欧美日韩二区三区| 国产精品久久一区二区三区不卡| 亚洲成人av电影在线| 又黄又爽又色视频| 日本一区二区三区免费乱视频 | 天堂视频中文在线| 1000精品久久久久久久久| 国产日韩av网站| 国产精品18久久久久久久久| 一区二区精品视频| 日本成人中文字幕在线视频| 免费电影一区| 日日骚欧美日韩| 日本一区二区三区免费看| 国产精品久久| 99爱精品视频| 亚洲国产91| 欧美专区一二三| 欧美私人啪啪vps| 亚洲aaa激情| 午夜精品毛片| 97在线资源站| 欧美日韩国产高清| 精品国产一区二区三区四区精华| 99伊人成综合| 亚洲图色在线| 99久久久国产精品免费蜜臀| 日韩av在线第一页| 国产欧美日韩在线视频| 97涩在线观看视频| 午夜国产不卡在线观看视频| 色就是色亚洲色图| 91精品国产综合久久久蜜臀粉嫩 | 中文字幕成人在线| 成年永久一区二区三区免费视频 | 亚洲成色777777女色窝| 蜜乳av一区| 亚洲欧美在线播放| 91亚洲无吗| 成人av在线网址| 国产亚洲福利| 中文字幕日韩精品无码内射| 久久综合九色综合欧美亚洲| gogo人体高清视频| 欧美日韩精品一区二区三区四区 | 国产一区日韩| 3d动漫精品啪啪一区二区三区免费 | 亚洲91精品| 欧美日韩免费精品| 成人av网站在线观看| youjizzxxxx18| jizzjizz国产精品喷水| 午夜精品久久久久久久久久| 最近中文字幕mv第三季歌词| 亚洲女人天堂色在线7777| 国产伦精品一区二区三区千人斩| 男人免费av| 国产欧美一区二区三区在线| 激情久久av一区av二区av三区| 欧美男男gaytwinkfreevideos| 日韩中文字幕组| 97在线精品视频| 欧美视频精品在线| 日韩欧美视频在线播放| 日韩精品一区二区三区四区五区 | 国产精品免费视频观看| 亚洲私人影吧| 午夜激情一区二区| 老司机在线视频二区| 中文字幕欧美日韩在线| 久久影视三级福利片| 国产一区不卡在线观看| 天天久久综合| 欧美激情一区二区三区成人 | 日韩精品一级中文字幕精品视频免费观看| 欧美一区二区三区爽大粗免费| 久久国产精品久久精品| 自拍偷拍欧美精品| 最新亚洲国产| 小泽玛利亚av在线| 久久全国免费视频| 久久久加勒比| 92看片淫黄大片看国产片| 国产综合色产在线精品| 德国一级在线视频| 欧美mv日韩mv国产网站| 日韩精品福利一区二区三区| 欧美午夜欧美| 一区二区三区中文字幕在线观看| 国产免费拔擦拔擦8x在线播放 | 国产91精品入口17c| 久久国产精品99久久人人澡| 日本免费高清视频| 中文字幕成人在线| 国产精品老牛| 中文字幕在线视频观看| 中文字幕在线视频日韩| 亚洲一区二区三区高清| 写真福利理论片在线播放| www.色综合| 日本中文字幕一区二区视频| 中文字幕在线中文字幕二区| 欧美插天视频在线播放| 牛牛澡牛牛爽一区二区| 亚洲校园欧美国产另类| 亚洲欧美中文字幕| 亚洲激情一区| av在线不卡播放| 欧美大片免费观看在线观看网站推荐| 欧美区一区二| av天天在线| 性色av一区二区咪爱| 粉嫩av亚洲一区二区图片| 日韩免费啪啪| 99热国产免费| 亚洲成av人片| 国产欧美一区| 中文字幕在线观看第三页| 最新国产成人av网站网址麻豆| 美腿丝袜在线亚洲一区| 麻豆视频在线| 国产专区一区二区| 欧美日韩在线视频首页| 香蕉视频一区二区三区| 91蝌蚪视频在线观看| 亚洲色图第三页| 国v精品久久久网| 亚洲爱爱视频| 日韩视频在线视频| 尤物yw午夜国产精品视频明星| 麻豆传媒一区二区三区| 91超碰国产在线| 三年中国中文在线观看免费播放| 精品sm在线观看| 久久精品国产精品青草| а√天堂中文在线资源8| 亚洲一二三区在线| 精品小视频在线| 国产一区二区精品久久99 | 欧美激情综合五月色丁香小说| 欧美一级做a| 看av免费毛片手机播放| 超薄丝袜一区二区| 中文字幕av一区二区三区免费看 | 电影k8一区二区三区久久| 中文字幕久热精品视频在线| 97国产在线视频| 亚洲欧美在线一区| 欧美日韩中文字幕一区二区| 国产精品久久久久桃色tv| 亚洲乱码精品| 夜鲁夜鲁夜鲁视频在线播放| 日韩精品成人av| 国产无遮挡在线视频免费观看| 欧美中文字幕在线观看视频| 国产农村妇女精品| 久久黄色影视| 精品国产鲁一鲁一区二区张丽| 免费日本一区二区三区视频| 99久久伊人精品影院| 欧美性欧美巨大黑白大战| 久久xxxx| 国产成人午夜性a一级毛片| 手机在线免费观看毛片| 国产精品视频播放| 在线电影院国产精品| 久久激情五月婷婷| 91嫩草国产线观看亚洲一区二区 | 久久97超碰国产精品超碰| 中国字幕a在线看韩国电影| 少妇黄色一级片| 国产高清在线精品一区二区三区| 日韩电影中文字幕| 亚洲精品老司机|