欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

RestTemplate自定義ErrorHandler方式

 更新時間:2021年08月19日 10:24:21   作者:zju611  
這篇文章主要介紹了RestTemplate自定義ErrorHandler方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教

RestTemplate自定義ErrorHandler

當(dāng)通過RestTemplate調(diào)用服務(wù)發(fā)生異常時,往往會返回400 Bad Request或500 internal error等錯誤信息。如果想捕捉服務(wù)本身拋出的異常信息,需要通過自行實現(xiàn)RestTemplate的ErrorHandler。

RestTemplate實例

可以通過調(diào)用setErrorHandler方法設(shè)置ErrorHandler,實現(xiàn)對請求響應(yīng)異常的判別和處理。自定義的ErrorHandler需實現(xiàn)ResponseErrorHandler接口,同時Spring boot也提供了默認(rèn)實現(xiàn)DefaultResponseErrorHandler,因此也可以通過繼承該類來實現(xiàn)自己的ErrorHandler。

getForObject和postForObject方法調(diào)用底層doExecute方法來執(zhí)行HTTP請求,通過Spring boot中doExecute方法可以看到RestTemplate在進(jìn)行HTTP請求時分成了

三個步驟:

1)創(chuàng)建請求,獲取響應(yīng);

2)判斷響應(yīng)是否異常,處理異常

3)將響應(yīng)消息體封裝為java對象

                Object varl4;
                // 1 創(chuàng)建請求,獲取響應(yīng)
                ClientHttpRequest request = this.createRequest(url, method);
                if (requestCallback != null) {
                    requestCallback.doWithRequest(request);
                }
                response = request.execute();
 
                // 2 判斷響應(yīng)是否存在異常,處理異常
                this.handleResponse(url, method, response);
 
                // 3 將響應(yīng)消息體封裝為java對象
                if (responseExtractor == null) {
                   resource = null;
                   return resource;
                }
                var14 = responseExtractor.extractData(response);

在handleResponse方法中對調(diào)用ErrorHandler來判斷響應(yīng)是否異常,并處理異常。這里需要注意的是,如果自定義ErrorHandler中的handlerError方法中獲取了response中body內(nèi)容就需要拋出異常,防止doExecute方法繼續(xù)執(zhí)行responseExtractor.extractData(response)語句導(dǎo)致response.body(類型為inputstream)被重復(fù)讀取。

    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);
        }
    }

學(xué)習(xí)了ErrorHandler在RestTemplate中的調(diào)用,開始實現(xiàn)自定義的ErrorHandler。首先創(chuàng)建自定義異常(由于ResponseErrorHandler中定義了handlerError方法拋出IOException,因此自定義的異常只能為RuntimeException)

public class MyException extends RuntimeException { 
    public MyException(String message){
        super(message);
    } 
    public MyException(String message, Throwable e){
        super(message + e.getLocalizedMessage());
    }
}

實現(xiàn)自定義ErrorHandler,一種思路是根據(jù)響應(yīng)消息體進(jìn)行相應(yīng)的異常處理策略,對于其他異常情況由父類DefaultResponseErrorHandler來進(jìn)行處理。

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 ,會直接拋出下面的異常。

如果想獲取原始的信息并進(jìn)一步處理會比較麻煩。

類似下面這種404、403響應(yīng)碼直接拋出異常并不是我們想要的

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 等常用方法最終都是調(diào)用 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));
        }
    }
}

如果想自己捕獲異常信息,自己處理異常的話可以通過實現(xiàn) ResponseErrorHandler 類來實現(xiàn)。其源碼如下:

public interface ResponseErrorHandler {
 
    // 標(biāo)示 Response 是否存在任何錯誤。實現(xiàn)類通常會檢查 Response 的 HttpStatus。
    boolean hasError(ClientHttpResponse var1) throws IOException;
 
    // 處理 Response 中的錯誤, 當(dāng) HasError 返回 true 時才調(diào)用此方法。
    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 中的報錯信息 而不拋出異常,可以通過下面的代碼實現(xiàn):

public class CustomErrorHandler implements ResponseErrorHandler { 
    @Override
    public boolean hasError(ClientHttpResponse response) throws IOException {
        return true;
    } 
    @Override
    public void handleError(ClientHttpResponse response) throws IOException {
 
    }
}

設(shè)置 RestTemplate 的異常處理類

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

輸出結(jié)果

{"code":404,"result":null,"message":"Resources not found"}

以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • Java實現(xiàn)消息轟炸的方法 附帶源碼

    Java實現(xiàn)消息轟炸的方法 附帶源碼

    這篇文章主要介紹了Java實現(xiàn)消息轟炸的方法 附帶源碼,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-04-04
  • 基于Java制作一個燈謎猜猜樂游戲

    基于Java制作一個燈謎猜猜樂游戲

    中秋佳節(jié),是我國傳統(tǒng)的重大節(jié)日之一,全國各地為了增強(qiáng)過節(jié)的氣氛,都有許多傳統(tǒng)的中秋活動,比如猜燈謎,所以本文就來用Java制作一個燈謎猜猜樂游戲,感興趣的可以了解下
    2023-09-09
  • Java實現(xiàn)自定義阻塞隊列

    Java實現(xiàn)自定義阻塞隊列

    這篇文章主要介紹了Java如何實現(xiàn)自定義阻塞隊列,幫助大家更好的理解和使用Java,感興趣的朋友可以了解下
    2020-10-10
  • Java版畫板的實現(xiàn)方法

    Java版畫板的實現(xiàn)方法

    這篇文章主要為大家詳細(xì)介紹了Java版畫板的實現(xiàn)方法,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-07-07
  • MyEclipse安裝JS代碼提示的教程(Spket插件)

    MyEclipse安裝JS代碼提示的教程(Spket插件)

    本篇文章主要介紹了MyEclipse安裝JS代碼提示的教程(Spket插件),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-12-12
  • spring中FactoryBean中的getObject()方法實例解析

    spring中FactoryBean中的getObject()方法實例解析

    這篇文章主要介紹了spring中FactoryBean中的getObject()方法實例解析,分享了相關(guān)代碼示例,小編覺得還是挺不錯的,具有一定借鑒價值,需要的朋友可以參考下
    2018-02-02
  • java 實例化類詳解及簡單實例

    java 實例化類詳解及簡單實例

    這篇文章主要介紹了java 實例化類詳解及簡單實例的相關(guān)資料,需要的朋友可以參考下
    2017-03-03
  • 啟動Solr提示Java版本低問題解決方案

    啟動Solr提示Java版本低問題解決方案

    這篇文章主要介紹了啟動Solr提示Java版本低問題解決方案,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2020-10-10
  • Java解析xml的四種方法匯總

    Java解析xml的四種方法匯總

    XML在不同的語言里解析方式都是一樣的,只不過實現(xiàn)的語法不同而已。java中基本的解析方式有四種,DOM解析、sax解析、JDOM解析和DOM4J解析,下面我們就來詳細(xì)探討下這四種方式
    2016-05-05
  • java中int、double、char等變量的取值范圍詳析

    java中int、double、char等變量的取值范圍詳析

    這篇文章主要給大家介紹了關(guān)于java中int、double、char等變量取值范圍的相關(guān)資料,每個變量都給出了詳細(xì)的實例代碼,對大家學(xué)習(xí)或者使用java具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2021-10-10

最新評論