RestTemplate自定義ErrorHandler
當通過RestTemplate調用服務發生異常時,往往會返回400 Bad Request或500 internal error等錯誤信息。如果想捕捉服務本身拋出的異常信息,需要通過自行實現RestTemplate的ErrorHandler。
RestTemplate實例
可以通過調用setErrorHandler方法設置ErrorHandler,實現對請求響應異常的判別和處理。自定義的ErrorHandler需實現ResponseErrorHandler接口,同時Spring boot也提供了默認實現DefaultResponseErrorHandler,因此也可以通過繼承該類來實現自己的ErrorHandler。
getForObject和postForObject方法調用底層doExecute方法來執行HTTP請求,通過Spring boot中doExecute方法可以看到RestTemplate在進行HTTP請求時分成了
三個步驟:
1)創建請求,獲取響應;
2)判斷響應是否異常,處理異常
3)將響應消息體封裝為java對象
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
Object varl4; // 1 創建請求,獲取響應 ClientHttpRequest request = this .createRequest(url, method); if (requestCallback != null ) { requestCallback.doWithRequest(request); } response = request.execute(); // 2 判斷響應是否存在異常,處理異常 this .handleResponse(url, method, response); // 3 將響應消息體封裝為java對象 if (responseExtractor == null ) { resource = null ; return resource; } var14 = responseExtractor.extractData(response); |
在handleResponse方法中對調用ErrorHandler來判斷響應是否異常,并處理異常。這里需要注意的是,如果自定義ErrorHandler中的handlerError方法中獲取了response中body內容就需要拋出異常,防止doExecute方法繼續執行responseExtractor.extractData(response)語句導致response.body(類型為inputstream)被重復讀取。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
protected void handleResponse(URI url, HttpMethod method, ClientHttpResponse response) throws IOException { ResponseErrorHandler errorHandler = this .getErrorHandler(); boolean hasError = errorHandler.hasError(response); if ( this .logger.isDebugEnabled()) { try { this .logger.debug(method.name() + " request for \"" + url + "\" resulted in " + response.getRawStatusCode() + " (" + response.getStatusText() + ")" + (hasError ? "; invoking error handler" : "" )); } catch (IOException var7) { ; } } if (hasError) { errorHandler.handleError(url, method, response); } } |
學習了ErrorHandler在RestTemplate中的調用,開始實現自定義的ErrorHandler。首先創建自定義異常(由于ResponseErrorHandler中定義了handlerError方法拋出IOException,因此自定義的異常只能為RuntimeException)
1
2
3
4
5
6
7
8
|
public class MyException extends RuntimeException { public MyException(String message){ super (message); } public MyException(String message, Throwable e){ super (message + e.getLocalizedMessage()); } } |
實現自定義ErrorHandler,一種思路是根據響應消息體進行相應的異常處理策略,對于其他異常情況由父類DefaultResponseErrorHandler來進行處理。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
public class MyErrorHandler extends DefaultResponseErrorHandler { @Override public boolean hasError(ClientHttpResponse response) throws IOException{ return super .hasError(response); } @Override public void handleError(ClientHttpResponse response) throws IOException{ Scanner scanner = new Scanner(response.getBody()).useDelimiter( "\\A" ); String stringResponse = scanner.hasNext() ? scanner.next() : "" ; if (stringResponse.matches( ".*XXX.*" )){ throw new MyException(stringResponse); } else { super .handleError(response); } } } |
SpringBoot 中使用 RestTemplate 自定義 異常處理,捕獲最原始的錯誤信息
一些 API 的報錯信息通過 Response 的 body返回。
使用 HttpClient 能正常獲取到 StatusCode 和 body 中的錯誤提示。然而使用 RestTemplate ,會直接拋出下面的異常。
如果想獲取原始的信息并進一步處理會比較麻煩。
類似下面這種404、403響應碼直接拋出異常并不是我們想要的
org.springframework.web.client.HttpClientErrorException: 404 null
at org.springframework.web.client.DefaultResponseErrorHandler.handleError(DefaultResponseErrorHandler.java:94)
at org.springframework.web.client.DefaultResponseErrorHandler.handleError(DefaultResponseErrorHandler.java:79)
at org.springframework.web.client.ResponseErrorHandler.handleError(ResponseErrorHandler.java:63)
at org.springframework.web.client.RestTemplate.handleResponse(RestTemplate.java:777)
at org.springframework.web.client.RestTemplate.doExecute(RestTemplate.java:730)
at org.springframework.web.client.RestTemplate.execute(RestTemplate.java:704)
at org.springframework.web.client.RestTemplate.exchange(RestTemplate.java:621)
RestTemplate 異常處理流程
下面看一下原因, RestTemplate 中的 getForObject, getForEntity 和 exchange 等常用方法最終都是調用 doExecute 方法。下面是 doExecute 方法源碼:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
|
public class RestTemplate extends InterceptingHttpAccessor implements RestOperations { private ResponseErrorHandler errorHandler; ...... @Nullable protected <T> T doExecute(URI url, @Nullable HttpMethod method, @Nullable RequestCallback requestCallback, @Nullable ResponseExtractor<T> responseExtractor) throws RestClientException { Assert.notNull(url, "'url' must not be null" ); Assert.notNull(method, "'method' must not be null" ); ClientHttpResponse response = null ; String resource; try { ClientHttpRequest request = this .createRequest(url, method); if (requestCallback != null ) { requestCallback.doWithRequest(request); } response = request.execute(); // 處理 Response this .handleResponse(url, method, response); if (responseExtractor != null ) { Object var14 = responseExtractor.extractData(response); return var14; } resource = null ; } catch (IOException var12) { resource = url.toString(); String query = url.getRawQuery(); resource = query != null ? resource.substring( 0 , resource.indexOf( 63 )) : resource; throw new ResourceAccessException( "I/O error on " + method.name() + " request for \"" + resource + "\": " + var12.getMessage(), var12); } finally { if (response != null ) { response.close(); } } return resource; } protected void handleResponse(URI url, HttpMethod method, ClientHttpResponse response) throws IOException { ResponseErrorHandler errorHandler = this .getErrorHandler(); boolean hasError = errorHandler.hasError(response); if ( this .logger.isDebugEnabled()) { try { this .logger.debug(method.name() + " request for \"" + url + "\" resulted in " + response.getRawStatusCode() + " (" + response.getStatusText() + ")" + (hasError ? "; invoking error handler" : "" )); } catch (IOException var7) { ; } } // 異常處理 if (hasError) { errorHandler.handleError(url, method, response); } } } |
從下面的代碼可以看出,DefaultResponseErrorHandler 捕獲并拋出了異常。
1
2
3
4
5
6
7
8
9
10
11
12
13
|
public class DefaultResponseErrorHandler implements ResponseErrorHandler { ... protected void handleError(ClientHttpResponse response, HttpStatus statusCode) throws IOException { switch (statusCode.series()) { case CLIENT_ERROR: throw new HttpClientErrorException(statusCode, response.getStatusText(), response.getHeaders(), this .getResponseBody(response), this .getCharset(response)); case SERVER_ERROR: throw new HttpServerErrorException(statusCode, response.getStatusText(), response.getHeaders(), this .getResponseBody(response), this .getCharset(response)); default : throw new UnknownHttpStatusCodeException(statusCode.value(), response.getStatusText(), response.getHeaders(), this .getResponseBody(response), this .getCharset(response)); } } } |
如果想自己捕獲異常信息,自己處理異常的話可以通過實現 ResponseErrorHandler 類來實現。其源碼如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
|
public interface ResponseErrorHandler { // 標示 Response 是否存在任何錯誤。實現類通常會檢查 Response 的 HttpStatus。 boolean hasError(ClientHttpResponse var1) throws IOException; // 處理 Response 中的錯誤, 當 HasError 返回 true 時才調用此方法。 void handleError(ClientHttpResponse var1) throws IOException; // handleError 的替代方案,提供訪問請求URL和HTTP方法的額外信息。 default void handleError(URI url, HttpMethod method, ClientHttpResponse response) throws IOException { this .handleError(response); } } |
自定義 RestTemplate 異常處理
如果想像 HttpClient 一樣直接從 Response 獲取 HttpStatus 和 body 中的報錯信息 而不拋出異常,可以通過下面的代碼實現:
1
2
3
4
5
6
7
8
9
10
|
public class CustomErrorHandler implements ResponseErrorHandler { @Override public boolean hasError(ClientHttpResponse response) throws IOException { return true ; } @Override public void handleError(ClientHttpResponse response) throws IOException { } } |
設置 RestTemplate 的異常處理類
1
2
3
|
restTemplate.setErrorHandler( new CustomErrorHandler()); ResponseEntity<String> response = restTemplate.exchange(uri, HttpMethod.GET, null , String. class ); System.out.println(response.getBody()); |
輸出結果
{"code":404,"result":null,"message":"Resources not found"}
以上為個人經驗,希望能給大家一個參考,也希望大家多多支持服務器之家。
原文鏈接:https://blog.csdn.net/zju611/article/details/80206207