解决RestTemplate 请求接收自定义400+ 或500+错误

作者:liulisha丶 时间:2023-12-12 00:25:20 

RestTemplate 请求接收自定义400+ 或500+错误

场景

当服务端自定义400错误返回体时,使用restTemplate 请求接收不到消息体。而我正想根据不同的错误信息做不同的操作。

原因

restTemplate 内置了自己的处理异常的方法ResponseErrorHandler去处理异常


protected void handleResponse(URI url, HttpMethod method, ClientHttpResponse response) throws IOException {
 ResponseErrorHandler errorHandler = getErrorHandler();
 boolean hasError = errorHandler.hasError(response);
 if (logger.isDebugEnabled()) {
  try {
   logger.debug(method.name() + " request for \"" + url + "\" resulted in " +
     response.getRawStatusCode() + " (" + response.getStatusText() + ")" +
     (hasError ? "; invoking error handler" : ""));
  }
  catch (IOException ex) {
   // ignore
  }
 }
 if (hasError) {
  errorHandler.handleError(response);
 }
}

当接收到CLIENT_ERROR 或 SERVER_ERROR 时,直接抛异常


@Override
public void handleError(ClientHttpResponse response) throws IOException {
 HttpStatus statusCode = getHttpStatusCode(response);
 switch (statusCode.series()) {
  case CLIENT_ERROR:
   throw new HttpClientErrorException(statusCode, response.getStatusText(),
     response.getHeaders(), getResponseBody(response), getCharset(response));
  case SERVER_ERROR:
   throw new HttpServerErrorException(statusCode, response.getStatusText(),
     response.getHeaders(), getResponseBody(response), getCharset(response));
  default:
   throw new RestClientException("Unknown status code [" + statusCode + "]");
 }
}

解决办法

自定义异常处理器,对响应的错误信息不进行处理


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

之后在bean 注入时,设置restTemplate 默认异常处理器为我们自定义的


@Bean
   public RestTemplate facePlusRestTemplate() {
       HttpComponentsClientHttpRequestFactory requestFactory = new HttpComponentsClientHttpRequestFactory();
       requestFactory.setConnectTimeout(300000);
       requestFactory.setReadTimeout(300000);
       RestTemplate restTemplate = new RestTemplate(requestFactory);
       restTemplate.getMessageConverters().set(1, new StringHttpMessageConverter(StandardCharsets.UTF_8));
       restTemplate.getMessageConverters().add(new FormHttpMessageConverter());
       restTemplate.setErrorHandler(new FacePlusThrowErrorHandler());
       return restTemplate;
   }

然后当我们使用restTemplate 时,设置restTemplate bean 名为注入时起的名字


@Resource(name = "facePlusRestTemplate")
private RestTemplate restTemplate;

最后从返回的ResponseEntity 中取body 属性,就可以取得服务端返回的消息体了

自定义RestTemplate的ResponseErrorHandler

Spring框架中的RestTemplate处理ClientHttpResponse的方式

直接看RestTemplate的源码

解决RestTemplate 请求接收自定义400+ 或500+错误 解决RestTemplate 请求接收自定义400+ 或500+错误

这里主要判断了是ClientError还是ServerError,即4xx或者是5xx。

如若是这两类状态码,会执行handleError()抛出异常

解决RestTemplate 请求接收自定义400+ 或500+错误

并不想让它抛异常

在一些业务场景下,或许我们并不想让它抛异常(即便我们可以捕获异常,额外做处理),那么就需要我们ResponseErrorHandler,并且重新定义一个RestTemplate对象使用该ErrorHandler。(简单实现如下)


@Configuration
public class CustomResponseErrorHandler implements ResponseErrorHandler {
   @Override
   public boolean hasError(ClientHttpResponse clientHttpResponse) throws IOException {
       return false;
   }
   @Override
   public void handleError(ClientHttpResponse clientHttpResponse) throws IOException {
   }
   @Bean("customRestTemplate")
   public RestTemplate customRestTemplate() {
       HttpComponentsClientHttpRequestFactory requestFactory = new HttpComponentsClientHttpRequestFactory();
       requestFactory.setConnectTimeout(300000);
       requestFactory.setReadTimeout(300000);
       RestTemplate restTemplate = new RestTemplate(requestFactory);
       restTemplate.getMessageConverters().set(1, new StringHttpMessageConverter(StandardCharsets.UTF_8));
       restTemplate.getMessageConverters().add(new FormHttpMessageConverter());
       restTemplate.setErrorHandler(new CustomResponseErrorHandler());
       return restTemplate;
   }
}

还需要加入一个maven依赖(具体原因查看maven依赖图)


<!-- https://mvnrepository.com/artifact/org.apache.httpcomponents/httpclient -->
<dependency>
   <groupId>org.apache.httpcomponents</groupId>
   <artifactId>httpclient</artifactId>
   <version>4.5.5</version>
</dependency>

在spring 中以IOC方式注入


@Resource(name = "customRestTemplate")
private RestTemplate restTemplate;

到这里已经可以正常使用。

无法使用IOC注入的场景下

还是参照CustomResponseErrorHandler中的customRestTemplate()去new对象吧……

来源:https://blog.csdn.net/qq_29369821/article/details/101351853

标签:RestTemplate,请求,400,500
0
投稿

猜你喜欢

  • Android中View的炸裂特效实现方法详解

    2022-07-25 17:59:28
  • java中为何重写equals时必须重写hashCode方法详解

    2022-10-24 01:44:39
  • 利用Java计算某个日期是星期几

    2023-11-17 05:49:42
  • C#根据http和ftp图片地址获取对应图片

    2023-06-06 02:00:23
  • 详解Android中Handler的使用方法

    2023-05-03 06:35:30
  • Android四大组件之Service详解

    2023-10-16 14:22:38
  • Android实现图片设置圆角形式

    2023-07-29 16:37:32
  • Minio与SpringBoot使用okhttp3问题解决

    2021-06-25 19:17:08
  • C#检测远程计算机端口是否打开的方法

    2023-12-11 08:55:31
  • Spring @Import注解的使用

    2022-03-26 16:38:43
  • SpringBoot 接口开发教程(httpclient客户端)

    2023-05-08 17:31:34
  • SSM框架整合之junit测试的方法

    2022-11-12 16:03:19
  • springmvc接收json串,转换为实体类List方法

    2023-07-28 12:40:00
  • C# WPF如何反射加载Geometry几何图形数据图标

    2021-09-06 11:39:08
  • 解析HikariCP一百行代码轻松掌握多线程

    2023-03-21 19:11:00
  • 全网最全SpringBoot集成swagger的详细教程

    2021-10-25 16:53:44
  • Mybatis MapperScannerConfigurer自动扫描Mapper接口生成代理注入到Spring的方法

    2023-04-17 11:57:25
  • 一文详解C#中重写(override)及覆盖(new)的区别

    2021-07-17 02:36:32
  • C#推送信息到APNs的方法

    2023-05-29 05:20:59
  • 详解Java的回调机制

    2023-07-27 07:17:43
  • asp之家 软件编程 m.aspxhome.com