本文是精讲RestTemplate第7篇,前篇的blog拜访地址如下:

  • 精讲RestTemplate第1篇-在Spring或非Spring环境下如何应用
  • 精讲RestTemplate第2篇-多种底层HTTP客户端类库的切换
  • 精讲RestTemplate第3篇-GET申请应用办法详解
  • 精讲RestTemplate第4篇-POST申请办法应用详解
  • 精讲RestTemplate第5篇-DELETE、PUT等申请办法应用详解
  • 精讲RestTemplate第6篇-文件上传下载与大文件流式下载

一、异常现象

在应用RestTemplate进行近程接口服务调用的时候,当申请的服务出现异常:超时、服务不存在等状况的时候(响应状态非200、而是400、500HTTP状态码),就会抛出如下异样:

该异样我是模仿进去的,将正确的申请服务地址由“/posts/1”改成“/postss/1”。服务不存在所以抛出404异样。

@Testpublic void testEntity() {   String url = "http://jsonplaceholder.typicode.com/postss/1";   ResponseEntity<String> responseEntity               = restTemplate.getForEntity(url, String.class);  //这行抛出异样   //上面两行代码执行不到   HttpStatus statusCode = responseEntity.getStatusCode(); // 获取响应码   System.out.println("HTTP 响应状态:" + statusCode);}

异样抛出之后,程序前面的代码就执行不到了,无奈进行前面的代码执行。理论的业务开发中,有的时候咱们更冀望的后果是:不论你服务端是超时了还是服务不存在,咱们都应该取得最终的申请后果(HTTP申请后果状态400、500),而不是取得一个抛出的异样。

二、源码解析-默认实现

首先我要说一个论断:RestTemplate申请后果异样是能够自定义解决的。在开始进行自定义的异样解决逻辑之前,咱们有必要看一下异样解决的默认实现。也就是:为什么会产生下面大节提到的景象?

  • ResponseErrorHandler是RestTemplate申请后果的异样处理器接口

    • 接口的第一个办法hasError用于判断HttpResponse是否是异样响应(通过状态码)
    • 接口的第二个办法handleError用于解决异样响应后果(非200状态码段)
  • DefaultResponseErrorHandler是ResponseErrorHandler的默认实现

所以咱们就来看看DefaultResponseErrorHandler是如何来解决异样响应的?从HttpResponse解析出Http StatusCode,如果状态码StatusCode为null,就抛出UnknownHttpStatusCodeException异样。

如果StatusCode存在,则解析出StatusCode的series,也就是状态码段(除了200段,其余全是异样状态码),解析规定是StatusCode/100取整。

public enum Series {   INFORMATIONAL(1),  // 1xx/100   SUCCESSFUL(2),  // 2xx/100   REDIRECTION(3), // 3xx/100   CLIENT_ERROR(4), // 4xx/100   ,客户端异样   SERVER_ERROR(5); // 5xx/100 ,服务端异样}

进一步针对客户端异样和服务端异样进行解决,解决的办法是抛出HttpClientErrorException。也就是第一大节呈现的异样的起因

三、RestTemplate自定义异样解决

所以咱们要实现自定义异样,实现ResponseErrorHandler 接口就能够。

public class MyRestErrorHandler implements ResponseErrorHandler {    /**     * 判断返回后果response是否是异样后果     * 次要是去查看response 的HTTP Status     * 仿造DefaultResponseErrorHandler实现即可     */    @Override    public boolean hasError(ClientHttpResponse response) throws IOException {        int rawStatusCode = response.getRawStatusCode();        HttpStatus statusCode = HttpStatus.resolve(rawStatusCode);        return (statusCode != null ? statusCode.isError(): hasError(rawStatusCode));    }    protected boolean hasError(int unknownStatusCode) {        HttpStatus.Series series = HttpStatus.Series.resolve(unknownStatusCode);        return (series == HttpStatus.Series.CLIENT_ERROR || series == HttpStatus.Series.SERVER_ERROR);    }     @Override    public void handleError(ClientHttpResponse response) throws IOException {        // 外面能够实现你本人遇到了Error进行正当的解决        //TODO 将接口申请的异样信息长久化    }}

将MyRestErrorHandler 在RestTemplate实例化的时候进行注册。参考: 《精讲RestTemplate第1篇-在Spring或非Spring环境下如何应用》 和 《精讲RestTemplate第2篇-多种底层HTTP客户端类库的切换》 进行实现

这时再去执行第一大节中的示例代码,就不会抛出异样。而是失去一个HTTP Status 404的后果。咱们能够依据这个后果,在程序中持续向下执行代码。

欢送关注我的博客,外面有很多精品合集

  • 本文转载注明出处(必须带连贯,不能只转文字):字母哥博客。

感觉对您有帮忙的话,帮我点赞、分享!您的反对是我不竭的创作能源! 。另外,笔者最近一段时间输入了如下的精品内容,期待您的关注。

  • 《手摸手教你学Spring Boot2.0》
  • 《Spring Security-JWT-OAuth2一本通》
  • 《实战前后端拆散RBAC权限管理系统》
  • 《实战SpringCloud微服务从青铜到王者》
  • 《VUE深入浅出系列》