解決RestTemplate 請求接收自定義400+ 或500+錯(cuò)誤
RestTemplate 請求接收自定義400+ 或500+錯(cuò)誤
場景
當(dāng)服務(wù)端自定義400錯(cuò)誤返回體時(shí),使用restTemplate 請求接收不到消息體。而我正想根據(jù)不同的錯(cuò)誤信息做不同的操作。
原因
restTemplate 內(nèi)置了自己的處理異常的方法ResponseErrorHandler去處理異常
protected void handleResponse(URI url, HttpMethod method, ClientHttpResponse response) throws IOException { ResponseErrorHandler errorHandler = getErrorHandler(); boolean hasError = errorHandler.hasError(response); if (logger.isDebugEnabled()) { try { logger.debug(method.name() + " request for \"" + url + "\" resulted in " + response.getRawStatusCode() + " (" + response.getStatusText() + ")" + (hasError ? "; invoking error handler" : "")); } catch (IOException ex) { // ignore } } if (hasError) { errorHandler.handleError(response); } }
當(dāng)接收到CLIENT_ERROR 或 SERVER_ERROR 時(shí),直接拋異常
@Override public void handleError(ClientHttpResponse response) throws IOException { HttpStatus statusCode = getHttpStatusCode(response); switch (statusCode.series()) { case CLIENT_ERROR: throw new HttpClientErrorException(statusCode, response.getStatusText(), response.getHeaders(), getResponseBody(response), getCharset(response)); case SERVER_ERROR: throw new HttpServerErrorException(statusCode, response.getStatusText(), response.getHeaders(), getResponseBody(response), getCharset(response)); default: throw new RestClientException("Unknown status code [" + statusCode + "]"); } }
解決辦法
自定義異常處理器,對響應(yīng)的錯(cuò)誤信息不進(jìn)行處理
public class FacePlusThrowErrorHandler implements ResponseErrorHandler { @Override public boolean hasError(ClientHttpResponse response) throws IOException { return false; } @Override public void handleError(ClientHttpResponse response) throws IOException { } }
之后在bean 注入時(shí),設(shè)置restTemplate 默認(rèn)異常處理器為我們自定義的
@Bean public RestTemplate facePlusRestTemplate() { HttpComponentsClientHttpRequestFactory requestFactory = new HttpComponentsClientHttpRequestFactory(); requestFactory.setConnectTimeout(300000); requestFactory.setReadTimeout(300000); RestTemplate restTemplate = new RestTemplate(requestFactory); restTemplate.getMessageConverters().set(1, new StringHttpMessageConverter(StandardCharsets.UTF_8)); restTemplate.getMessageConverters().add(new FormHttpMessageConverter()); restTemplate.setErrorHandler(new FacePlusThrowErrorHandler()); return restTemplate; }
然后當(dāng)我們使用restTemplate 時(shí),設(shè)置restTemplate bean 名為注入時(shí)起的名字
@Resource(name = "facePlusRestTemplate") private RestTemplate restTemplate;
最后從返回的ResponseEntity 中取body 屬性,就可以取得服務(wù)端返回的消息體了
自定義RestTemplate的ResponseErrorHandler
Spring框架中的RestTemplate處理ClientHttpResponse的方式
直接看RestTemplate的源碼
這里主要判斷了是ClientError還是ServerError,即4xx或者是5xx。
如若是這兩類狀態(tài)碼,會執(zhí)行handleError()拋出異常
并不想讓它拋異常
在一些業(yè)務(wù)場景下,或許我們并不想讓它拋異常(即便我們可以捕獲異常,額外做處理),那么就需要我們ResponseErrorHandler,并且重新定義一個(gè)RestTemplate對象使用該ErrorHandler。(簡單實(shí)現(xiàn)如下)
@Configuration public class CustomResponseErrorHandler implements ResponseErrorHandler { @Override public boolean hasError(ClientHttpResponse clientHttpResponse) throws IOException { return false; } @Override public void handleError(ClientHttpResponse clientHttpResponse) throws IOException { } @Bean("customRestTemplate") public RestTemplate customRestTemplate() { HttpComponentsClientHttpRequestFactory requestFactory = new HttpComponentsClientHttpRequestFactory(); requestFactory.setConnectTimeout(300000); requestFactory.setReadTimeout(300000); RestTemplate restTemplate = new RestTemplate(requestFactory); restTemplate.getMessageConverters().set(1, new StringHttpMessageConverter(StandardCharsets.UTF_8)); restTemplate.getMessageConverters().add(new FormHttpMessageConverter()); restTemplate.setErrorHandler(new CustomResponseErrorHandler()); return restTemplate; } }
還需要加入一個(gè)maven依賴(具體原因查看maven依賴圖)
<!-- https://mvnrepository.com/artifact/org.apache.httpcomponents/httpclient --> <dependency> <groupId>org.apache.httpcomponents</groupId> <artifactId>httpclient</artifactId> <version>4.5.5</version> </dependency>
在spring 中以IOC方式注入
@Resource(name = "customRestTemplate") private RestTemplate restTemplate;
到這里已經(jīng)可以正常使用。
無法使用IOC注入的場景下
還是參照CustomResponseErrorHandler中的customRestTemplate()去new對象吧……
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Java中SSM框架實(shí)現(xiàn)增刪改查功能代碼詳解
這篇文章主要介紹了Java中SSM框架實(shí)現(xiàn)增刪改查功能代碼詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-07-07基于Restful接口調(diào)用方法總結(jié)(超詳細(xì))
下面小編就為大家?guī)硪黄赗estful接口調(diào)用方法總結(jié)(超詳細(xì))。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-08-08elasticsearch數(shù)據(jù)信息索引操作action?support示例分析
這篇文章主要為大家介紹了elasticsearch數(shù)據(jù)信息索引操作action?support示例分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-04-04IDEA類與方法注釋模板設(shè)置圖文教程(非常詳細(xì))
IDEA自帶的注釋模板不是太好用,我本人到網(wǎng)上搜集了很多資料系統(tǒng)的整理了一下制作了一份比較完整的模板來分享給大家,下面這篇文章主要給大家介紹了關(guān)于IDEA類與方法注釋模板設(shè)置的相關(guān)資料,需要的朋友可以參考下2022-09-09SpringBoot Pom文件依賴及Starter啟動(dòng)器詳細(xì)介紹
這篇文章主要介紹了SpringBoot Pom文件的依賴與starter啟動(dòng)器的作用,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-09-09