天天看點

【RestTemplate】自定義 RestTemplate 異常處理自定義 RestTemplate 異常處理

文章目錄

  • 自定義 RestTemplate 異常處理
    • RestTemplate 異常處理流程
    • 自定義 RestTemplate 異常處理
    • 參考

自定義 RestTemplate 異常處理

一些

API

的報錯資訊通過

Response

body

傳回。使用

HttpClient

能正常擷取到

StatusCode

body

中的錯誤提示。然而使用

RestTemplate

,會直接抛出下面的異常。如果想擷取原始的資訊并進一步處理會比較麻煩。

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

方法源碼:

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

捕獲并抛出了異常。

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

類來實作。其源碼如下:

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

中的報錯資訊 而不抛出異常,可以通過下面的代碼實作:

public class CustomErrorHandler implements ResponseErrorHandler {

    @Override
    public boolean hasError(ClientHttpResponse response) throws IOException {
        return true;
    }

    @Override
    public void handleError(ClientHttpResponse response) throws IOException {

    }
}
           

設定

RestTemplate

的異常處理類

restTemplate.setErrorHandler(new CustomErrorHandler());
ResponseEntity<String> response = restTemplate.exchange(uri, HttpMethod.GET, null, String.class);
System.out.println(response.getBody());
           

輸出結果

參考

  • 參考文檔
  • 參考源碼
  • 本文源碼

本文轉載自Silo Blog平台Jekyll的自定義 RestTemplate 異常處理一文。