【java后端】restTemplate异常处理

  • 记录一下今天遇到的问题,和尝试解决过程中的经历 -

在与其他微服务接口进行联调的过程中,因为我这边依赖的接口在正常和异常两种情况下,返回的字段不一样,导致我这边通过restTemplate发送请求时,封装的返回对象不能够灵活适配接口返回对象的字段,于是在服务异常返回时,我这边处理不了。
于是查询解决这个问题的方法,发现可以自己定义一个ResponseErrorHandler的实现类,对异常情况进行处理。

依赖接口返回参数示例

正常情况:

{
    "library_id": "36"
}

异常情况:

{
    "error_code": 8008,
    "error_msg": "",
    "error_type": "library_name is not string"
}

由于依赖接口正常和异常情况下http状态码都是200,导致不能通过状态码来判断哪种情况是异常情况,只能通过取body中的值来校验,写了如下FaceResponseErrorHandler类。
ResponseErrorHandler的实现类:

public class FaceResponseErrorHandler implements ResponseErrorHandler {

    /** 对response进行判断,如果是异常情况,返回true */
    @Override
    public boolean hasError(ClientHttpResponse response) throws IOException {
        Map map = JSON.parseObject(convertStreamToString(response.getBody()));
        return map.containsKey("error_code");
    }

    /** 异常情况时的处理方法 */
    @Override
    public void handleError(ClientHttpResponse response) {
        throw new LibraryException(FaceErrorCode.IA_LIBRARY_ERROR, FaceResponseErrorHandler.class);
    }

    /**
     * InputStream转成String
     * @param is InputStream
     * @return String
     */
    private String convertStreamToString(InputStream is) {

        BufferedReader reader = new BufferedReader(new InputStreamReader(is));
        StringBuilder sb = new StringBuilder();

        String line;
        try {
            while ((line = reader.readLine()) != null) {
                sb.append(line);
            }
        } catch (IOException e) {
            throw BaseException.of(FaceErrorCode.IA_LIBRARY_ERROR.of());
        } finally {
            is.close();
        }

        return sb.toString();
    }
}

FaceResponseErrorHandler调用:

	restTemplate.setErrorHandler(new FaceResponseErrorHandler());
    Object res = restTemplate.postForObject(host + IAConstant.LIBRARY_MANAGER, addReqDO, Object.class);

测试后发现,这种方式存在异常报错或获取到的res为Null的问题,原因是在FaceResponseErrorHandler类中convertStreamToString()方法里读取数据并转化之后,在finally里面将流关闭,导致后续restTemplate原生方法在处理返回值的时候报流已关闭的错误,即使将finally代码去除,由于在convertStreamToString()方法读取了InputStream对象里面的内容,而InputStream只能读取一次,在InputStream读取之后response的body就空了,后面在获取结果时得到是null,于是这种方法行不通,除非还是通过http状态码来进行判断,但目前状态码条件不满足,最后还是弃用了FaceResponseErrorHandler,又将restTemplate的返回对象定义为Object类型。在debug调试过程中意外发现返回的object对象其实是LinkedHashMap类型的,于是采用了一个比较low的方法来解决这个问题,即将res进行了强转成LinkedHashMap来获取值,然后对数据进行处理:

/** 强转数据并处理 */
private IALibraryRes getIALibraryRes(Object res) {
    IALibraryRes iaLibraryRes = new IALibraryRes();
    if (res != null) {
        if (res instanceof LinkedHashMap) {
            LinkedHashMap resMap = (LinkedHashMap) res;
            if (resMap.containsKey("error_code")) {
                for (Object key : resMap.keySet()) {
                    LOGGER.info(String.valueOf(key) + ": " + String.valueOf(resMap.get(key)));
                }
                throw new LibraryException(FaceErrorCode.IA_LIBRARY_ERROR, IALibraryServiceImpl.class);
            } else {
                iaLibraryRes.setLibraryId(String.valueOf(resMap.get("library_id")));
            }
        } else {
            throw new LibraryException(FaceErrorCode.IA_LIBRARY_ERROR, IALibraryServiceImpl.class);
        }
    } else {
        throw new LibraryException(FaceErrorCode.IA_LIBRARY_ERROR, IALibraryServiceImpl.class);
    }
    return iaLibraryRes;
}

参考网址:
https://blog.csdn.net/achang07/article/details/80549741
https://my.oschina.net/ChenGuop/blog/1581759?tdsourcetag=s_pctim_aiomsg
https://www.cnblogs.com/fzll/p/3400558.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
对于使用 RestTemplate 发起 HTTP 请求时可能出现的异常,可以进行以下异常处理: 1. HttpClientErrorException:当响应的状态码为 4xx(如 404 Not Found)时,会抛出此异常。可以通过 `getStatusCode()` 方法获取状态码,`getResponseBodyAsString()` 获取错误响应的内容。 ```java try { ResponseEntity<String> response = restTemplate.exchange(url, HttpMethod.GET, null, String.class); // 处理成功响应 } catch (HttpClientErrorException e) { if (e.getStatusCode() == HttpStatus.NOT_FOUND) { // 处理 404 错误 } else { // 处理其他 4xx 错误 } } ``` 2. HttpServerErrorException:当响应的状态码为 5xx(如 500 Internal Server Error)时,会抛出此异常。 ```java try { ResponseEntity<String> response = restTemplate.exchange(url, HttpMethod.GET, null, String.class); // 处理成功响应 } catch (HttpServerErrorException e) { if (e.getStatusCode() == HttpStatus.INTERNAL_SERVER_ERROR) { // 处理 500 错误 } else { // 处理其他 5xx 错误 } } ``` 3. ResourceAccessException:在网络连接或请求超时等异常情况下,会抛出此异常。可以通过捕获该异常来处理网络异常情况。 ```java try { ResponseEntity<String> response = restTemplate.exchange(url, HttpMethod.GET, null, String.class); // 处理成功响应 } catch (ResourceAccessException e) { // 处理网络异常 } ``` 除了上述异常处理,还可以通过使用 `ResponseErrorHandler` 接口来自定义异常处理。通过实现该接口,可以针对不同的响应状态码进行自定义处理。 ```java restTemplate.setErrorHandler(new DefaultResponseErrorHandler() { @Override public void handleError(ClientHttpResponse response) throws IOException { if (response.getStatusCode() == HttpStatus.NOT_FOUND) { // 处理 404 错误 } else if (response.getStatusCode() == HttpStatus.INTERNAL_SERVER_ERROR) { // 处理 500 错误 } else { super.handleError(response); } } }); ``` 这样,在发起 HTTP 请求时,如果发生错误,就会根据自定义的异常处理进行处理。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值