Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 453|回复: 0

SpringWeb 系列教程 RestTemplate 4xx/5xx 异常信息捕获

[复制链接]
  • TA的每日心情
    奋斗
    2024-11-24 15:47
  • 签到天数: 804 天

    [LV.10]以坛为家III

    2053

    主题

    2111

    帖子

    72万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    726782
    发表于 2021-8-28 09:48:26 | 显示全部楼层 |阅读模式

    200104-SpringWeb 系列教程 RestTemplate 4xx/5xx 异常信息捕获

    近期使用 RestTemplate 访问外部资源时,发现一个有意思的问题。因为权限校验失败,对方返回的 401 的 http code,此外返回数据中也会包含一些异常提示信息;然而在使用 RestTemplate 访问时,却是直接抛了如下提示 401 的异常,并不能拿到提示信息

    那么 RestTemplate 如果希望可以获取到非 200 状态码返回数据时,可以怎么操作呢?

    I. 异常捕获

    1. 问题分析

    RestTemplate 的异常处理,是借助org.springframework.web.client.ResponseErrorHandler来做的,先看一下两个核心方法

    • 下面代码来自 spring-web.5.0.7.RELEASE 版本
    public interface ResponseErrorHandler {
      // 判断是否有异常
    	boolean hasError(ClientHttpResponse response) throws IOException;
      // 如果有问题,进入这个方法,处理问题
    	void handleError(ClientHttpResponse response) throws IOException;
    }
    

    简单来讲,当 RestTemplate 发出请求,获取到对方相应之后,会交给ResponseErrorHandler来判断一下,返回结果是否 ok

    因此接下来将目标瞄准到 RestTemplate 默认的异常处理器: org.springframework.web.client.DefaultResponseErrorHandler

    a. 判定返回结果是否 ok

    从源码上看,主要是根据返回的 http code 来判断是否 ok

    // 根据返回的http code判断有没有问题
    @Override
    public boolean hasError(ClientHttpResponse response) throws IOException {
    	HttpStatus statusCode = HttpStatus.resolve(response.getRawStatusCode());
    	return (statusCode != null && hasError(statusCode));
    }
    
    // 具体的判定逻辑,简单来讲,就是返回的http code是标准的4xx, 5xx,那么就认为有问题了
    protected boolean hasError(HttpStatus statusCode) {
    	return (statusCode.series() == HttpStatus.Series.CLIENT_ERROR ||
    			statusCode.series() == HttpStatus.Series.SERVER_ERROR);
    }
    

    请注意上面的实现,自定义的某些 http code 是不会被认为是异常的,因为无法转换为对应的HttpStatus (后面实例进行说明)

    b. 异常处理

    当上面的 hasError 返回 ture 的时候,就会进入异常处理逻辑

    @Override
    public void handleError(ClientHttpResponse response) throws IOException {
    	HttpStatus statusCode = HttpStatus.resolve(response.getRawStatusCode());
    	if (statusCode == null) {
    		throw new UnknownHttpStatusCodeException(response.getRawStatusCode(), response.getStatusText(),
    				response.getHeaders(), getResponseBody(response), getCharset(response));
    	}
    	handleError(response, statusCode);
    }
    
    protected void handleError(ClientHttpResponse response, HttpStatus statusCode) throws IOException {
    	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 UnknownHttpStatusCodeException(statusCode.value(), response.getStatusText(),
    					response.getHeaders(), getResponseBody(response), getCharset(response));
    	}
    }
    

    从上面也可以看到,异常处理逻辑很简单,直接抛异常

    2. 异常捕获

    定位到生面的问题之后,再想解决问题就相对简单了,自定义一个异常处理类,不管状态码返回是啥,全都认为正常即可

    RestTemplate restTemplate = new RestTemplate();
    restTemplate.setErrorHandler(new DefaultResponseErrorHandler(){
        @Override
        protected boolean hasError(HttpStatus statusCode) {
            return super.hasError(statusCode);
        }
    
        @Override
        public void handleError(ClientHttpResponse response) throws IOException {
        }
    });
    

    3. 实测

    首先写两个结果,返回的 http 状态码非 200;针对返回非 200 状态码的 case,有多种写法,下面演示两种常见的

    @RestController
    public class HelloRest {
      @GetMapping("401")
      public ResponseEntity<String> _401(HttpServletResponse response) {
          ResponseEntity<String> ans =
                  new ResponseEntity<>("{\"code\": 401, \"msg\": \"some error!\"}", HttpStatus.UNAUTHORIZED);
          return ans;
      }
    
      @GetMapping("525")
      public String _525(HttpServletResponse response) {
          response.setStatus(525);
          return "{\"code\": 525, \"msg\": \"自定义错误码!\"}";
      }
    }
    

    首先来看一下自定义的 525 和标准的 401 http code,直接通过RestTemplate访问的 case

    @Test
    public void testCode() {
        RestTemplate restTemplate = new RestTemplate();
        HttpEntity<String> ans = restTemplate.getForEntity("http://127.0.0.1:8080/525", String.class);
        System.out.println(ans);
    
        ans = restTemplate.getForEntity("http://127.0.0.1:8080/401", String.class);
        System.out.println(ans);
    }
    

    从上面的输出结果也可以看出来,非标准 http code 不会抛异常(原因上面有分析),接下来看一下即便是标准的 http code 也不希望抛异常的 case

    @Test
    public void testSend() {
        String url = "http://127.0.0.1:8080/401";
        RestTemplate restTemplate = new RestTemplate();
        restTemplate.setErrorHandler(new DefaultResponseErrorHandler(){
            @Override
            protected boolean hasError(HttpStatus statusCode) {
                return super.hasError(statusCode);
            }
    
            @Override
            public void handleError(ClientHttpResponse response) throws IOException {
            }
        });
        HttpEntity<String> ans = restTemplate.getForEntity(url, String.class);
        System.out.println(ans);
    }
    

    II. 其他

    0. 项目

    1. 一灰灰 Blog

    尽信书则不如,以上内容,纯属一家之言,因个人能力有限,难免有疏漏和错误之处,如发现 bug 或者有更好的建议,欢迎批评指正,不吝感激

    下面一灰灰的个人博客,记录所有学习和工作中的博文,欢迎大家前去逛逛

    一灰灰blog

    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2025-1-4 03:18 , Processed in 0.058498 second(s), 29 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表