本篇內容主要講解“使用RestTemplate時報錯RestClientException怎么解決”,感興趣的朋友不妨來看看。本文介紹的方法操作簡單快捷,實用性強。下面就讓小編來帶大家學習“使用RestTemplate時報錯RestClientException怎么解決”吧!
使用RestTemplate時報錯RestClientException
這是自己封裝的一個發送請求的方法
這是自定義的一個http信息Converter
RestTemplate的錯誤處理
問題描述
ErrorHandler
解決辦法
public Map<String, Object> sendRequest(Map<String, Object> body,String sessionId,String url) { RestTemplate restTemplate = new RestTemplate(); restTemplate.getMessageConverters().add(new BdMappingJackson2HttpMessageConverter()); Map<String, Object> map = new HashMap<>(); try { ParameterizedTypeReference<Map<String, Object>> typeRef = new ParameterizedTypeReference<Map<String, Object>>() { }; HttpHeaders headers = new HttpHeaders(); headers.setContentType(MediaType.APPLICATION_OCTET_STREAM); headers.set("cookie", "SESSION="+sessionId); HttpEntity<Map<String, Object>> requestEntity = new HttpEntity<>(body,headers); ResponseEntity<Map<String, Object>> responseEntity = restTemplate.exchange(url, HttpMethod.POST, requestEntity,typeRef); map = responseEntity.getBody(); log.info(map.toString()); } catch (HttpStatusCodeException e) { log.error(e.getResponseBodyAsString(), e); map = JsonUtil.toMap(e.getResponseBodyAsString()); } return map; }
public class BdMappingJackson2HttpMessageConverter extends MappingJackson2HttpMessageConverter { public BdMappingJackson2HttpMessageConverter(){ List<MediaType> mediaTypes = new ArrayList<>(); mediaTypes.add(MediaType.TEXT_HTML); mediaTypes.add(MediaType.APPLICATION_OCTET_STREAM); setSupportedMediaTypes(mediaTypes); } }
我遇到的第一個問題是這樣的
Could not write request: no suitable HttpMessageConverter found for request type [java.util.HashMap] and content type [application/octet-stream]
意思大概是無法寫入請求:找不到適用于請求類型[java.util.HashMap]和內容類型[application/octet stream]的HttpMessageConverter
它默認只處理application/json:JSON數據格式,這個二進制數據流格式不支持,所以我在那個自定義的Converter加上去了。
緊接著又遇到第二個問題
Could not extract response: no suitable HttpMessageConverter found for response type [java.util.Map<java.lang.String, java.lang.Object>] and content type [text/xml;charset=UTF-8]
無法提取響應:找不到適合于響應類型[java.util.Map<java.lang.String,java.lang.Object>]和內容類型[textml/charset=UTF-8]的HttpMessageConverter
這是那邊回調過來的Content-Type類型是text/xml它同樣解析不了,所以我把這個也加上去了,至此就ok了。
常見的Content-Type類型:
text/html
:HTML格式
text/plain
:純文本格式
image/png
:png圖片格式
application/json
:JSON數據格式
application/octet-stream
:二進制流數據
application/x-www-form-urlencoded
:表單中默認的encType,表單數據被編碼為key/value格式發送到服務器
multipart/form-data
:需要在表單中進行文件上傳時,就需要使用該格式
我們的項目屬于微服務架構,兩個基礎的服務分別是網關和認證鑒權服務。
在前端訪問后臺服務的時候,都經過網關轉發,轉發之前會進行鑒權認證,根據鑒權結果判斷是否可以進行相應的請求轉發。
認證鑒權服務中,判斷Token對應的人員是否有相應的權限,如果沒有權限,返回401狀態碼并在響應體中傳回錯誤信息。
網關與鑒權服務之間的服務調用通過RestTemplate進行(可以考慮轉為Feign做聲明式的服務調用),然而如果鑒權服務返回401的情況下,網關服務直接報出HttpClientErrorException,讓人一頭霧水。
實際上,答案都在源碼中,看一下RestTemplate的源碼就知曉了。
在RestTemplate中,有一個成員變量ResponseErrorHandler。
ResponseErrorHandler是一個接口,包括兩個方法:
public interface ResponseErrorHandler { boolean hasError(ClientHttpResponse var1) throws IOException; void handleError(ClientHttpResponse var1) throws IOException; }
這個接口有一個默認實現DefaultResponseErrorHandler。該方法中,判斷是否發生error的方法hasError最終調用的方法如下:
protected boolean hasError(HttpStatus statusCode) { return statusCode.series() == Series.CLIENT_ERROR || statusCode.series() == Series.SERVER_ERROR; }
很明顯,根據響應狀態嗎為4xx或者5xx來認定發生了錯誤。而錯誤處理在handleError中:
public void handleError(ClientHttpResponse response) throws IOException { HttpStatus statusCode = this.getHttpStatusCode(response); switch(null.$SwitchMap$org$springframework$http$HttpStatus$Series[statusCode.series().ordinal()]) { case 1: throw new HttpClientErrorException(statusCode, response.getStatusText(), response.getHeaders(), this.getResponseBody(response), this.getCharset(response)); case 2: throw new HttpServerErrorException(statusCode, response.getStatusText(), response.getHeaders(), this.getResponseBody(response), this.getCharset(response)); default: throw new RestClientException("Unknown status code [" + statusCode + "]"); } }
可見,4xx的狀態嗎會拋出HttpClientErrorException;5xx的狀態碼會拋出HttpServerErrorException。這也就是我們一開始遇到的問題的原因所在了。而在handleError中,執行了response.getBody(),這就導致我們后續獲取不到響應體了,如果要獲取的話,需要進行自定義相關處理。
如果RestTemplate的應用場景比較統一,可以自定義ResponseErorHandler(派生自DefaultResponseErrorHandler)來接管錯誤處理,進行自己想要的處理。
而我們的網關中,對于頁面跳轉類的請求和Rest API類的請求,處理辦法顯然是不一樣的。所以最終處理是catch異常,然后進行重定向的處理操作。
到此,相信大家對“使用RestTemplate時報錯RestClientException怎么解決”有了更深的了解,不妨來實際操作一番吧!這里是億速云網站,更多相關內容可以進入相關頻道進行查詢,關注我們,繼續學習!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。