前言

项目中有个模板的功能是可以模仿postman进行http请求。 之前使用过httpClient,这次就来试试看更加模板化的RestTemplate吧~ 

关于RestTemplate的教程网上有很多,但大多都是秉着授人以渔的心态写的,也就是说在实战方面写得并不是很详细。这次我就借着模块开发的机会,来分享一下我是怎么编写http请求工具类的吧。

首先,要使用RestTemplate的exchange方法作为整个工具类的基础,借助它的可自定义配置,我们可以实现 使用任意格式的请求参数 和 返回任意格式的结果。这对于一个工具类来说是非常重要的。

需要注意的是,在RestTemplate中使用post方式请求 + 使用formdata格式传递body参数时,不能使用Map,需要要使用MultiValueMap


实战环节

代码如下(示例):

/**
 * RestTemplateUtils 【http请求工具类】
 * 1. sendGet() sendPost() 是正常http请求
 * <br>
 * 2. sendGetByByte() sendPostByByte() 是用来下载字节流的
 * <br>
 * 3. 支持的请求方式:<br>
 * <b>
 * post请求 -json传参<br>
 * post请求 -formData传参<br>
 * get请求 -占位符传参<br>
 * get请求 -url拼接<br>
 * </b>
 *
 * @author xurenyi
 * @since 2023/3/29 17:36
 **/
public class RestTemplateUtils {

    private static Logger logger = LoggerFactory.getLogger("RestTemplateUtils");
    private volatile static RestTemplate restTemplate;

    /**
     * GET请求
     *
     * @param url
     * @param returnDataType 返回数据的格式 JSONObject|JSONArray.class|String.class|Map.class|ArrayList.class,可为空
     * @param requestHeaders 请求头,可为空
     * @param requestParams  求参数,可为空
     * @return 结果
     */
    public static Object sendGet(String url,
                                 @Nullable Class<?> returnDataType,
                                 @Nullable Map<String, Object> requestHeaders,
                                 @Nullable Map<String, Object> requestParams) {
        return httpRequest(url, HttpMethod.GET, requestHeaders, requestParams, null, returnDataType, JSONObject.class);
    }


    /**
     * POST请求
     *
     * @param url
     * @param returnDataType 返回数据的格式 JSONObject|JSONArray.class|String.class|Map.class|ArrayList.class,可为空
     * @param requestHeaders 请求头,可为空
     * @param requestParams  请求参数,可为空
     * @param requestBody    请求体,可为空
     * @param sendDataType   请求体格式 json|jsonArray
     * @return 结果
     */
    public static Object sendPost(String url,
                                  @Nullable Class<?> returnDataType,
                                  @Nullable Map<String, Object> requestHeaders,
                                  @Nullable Map<String, Object> requestParams,
                                  @Nullable String requestBody,
                                  Class<?> sendDataType) {
        return httpRequest(url, HttpMethod.POST, requestHeaders, requestParams, requestBody, returnDataType, sendDataType);
    }


    /**
     * POST请求 字节流
     *
     * @param url
     * @param response       http响应消息,可为空
     * @param requestHeaders 请求头,可为空
     * @param requestParams  请求参数,可为空
     * @param requestBody    请求体,可为空
     * @param sendDataType   请求体格式 json|jsonArray
     **/
    public static void sendPostByByte(String url, HttpServletResponse response,
                                      @Nullable Map<String, Object> requestHeaders,
                                      @Nullable Map<String, Object> requestParams,
                                      @Nullable String requestBody,
                                      Class<?> sendDataType) {
        download(url, HttpMethod.POST, requestHeaders, requestParams, requestBody, response, false, sendDataType);
    }

    /**
     * GET请求 字节流
     *
     * @param url
     * @param response       http响应消息,可为空
     * @param requestHeaders 请求头,可为空
     * @param requestParams  请求参数,可为空
     **/
    public static void sendGetByByte(String url, HttpServletResponse response,
                                     @Nullable Map<String, Object> requestHeaders,
                                     @Nullable Map<String, Object> requestParams) {
        download(url, HttpMethod.GET, requestHeaders, requestParams, null, response, false, JSONObject.class);
    }

    /**
     * 发送http请求
     *
     * @param url
     * @param method
     * @param requestHeaders
     * @param requestParams
     * @param requestBody
     * @param returnDataType
     * @return
     */
    public static <T> Object httpRequest(String url, HttpMethod method,
                                         @Nullable Map<String, Object> requestHeaders,
                                         @Nullable Map<String, Object> requestParams,
                                         @Nullable String requestBody,
                                         @Nullable Class<?> returnDataType,
                                         Class<?> sendDataType) {

        Map<String, Object> params = new HashMap<>();
        HttpHeaders headers = new HttpHeaders();
        // 1. post请求 -json传参
        // 2. post请求 -formData传参
        // 3. get请求 -占位符传参
        // 4. get请求 -url拼接

        // 第一步 构建httpEntity
        HttpEntity<?> entity = buildHttpEntity(sendDataType, method, requestHeaders, requestParams, requestBody, params, headers);
        // 第二步 单例调用restTemplate
        RestTemplate restTemplate = getSingleRestTemplate();
        // 第三步 初始化一个返回值
        Object r = null;
        // 第四步 执行http请求,并根据情况给返回值赋值
        try {
            Class<?> resultClass = String.class;
            if (returnDataType != null) {
                resultClass = returnDataType;
                r = handleHttpRequest(restTemplate, resultClass, url, method, params, entity);
            } else {
                handleHttpRequest(restTemplate, resultClass, url, method, params, entity);
            }

        } catch (RestClientException e) {
            logger.error("远程调用接口异常{}", e);
        }
        return r;
    }

    /**
     * 下载
     *
     * @param url
     * @param method
     * @param requestHeaders
     * @param requestParams
     * @param requestBody
     * @param response
     * @param isRespBodyOnly 是否是只需要响应体,默认true,true:只需要响应体  false:同时返回响应状态 响应头  响应体
     */
    public static void download(String url, HttpMethod method,
                                @Nullable Map<String, Object> requestHeaders,
                                @Nullable Map<String, Object> requestParams,
                                @Nullable String requestBody,
                                HttpServletResponse response,
                                @Nullable Boolean isRespBodyOnly,
                                Class<?> sendDataType) {
        if (Objects.isNull(isRespBodyOnly)) {
            isRespBodyOnly = true;
        }
        HttpHeaders headers = new HttpHeaders();
        Map<String, Object> params = new HashMap<>();

        // 第一步 构建httpEntity
        HttpEntity<?> entity = buildHttpEntity(sendDataType, method, requestHeaders, requestParams, requestBody, params, headers);
        // 第二步 单例调用restTemplate
        RestTemplate restTemplate = getSingleRestTemplate();
        // 第三步 执行http请求,并返回字节流
        handleHttpRequest(restTemplate, url, method, response, isRespBodyOnly, params, entity);
    }

    /**
     * 构建httpEntity
     *
     * @param method
     * @param requestHeaders
     * @param requestParams
     * @param requestBody
     * @param params
     * @param headers
     * @return
     */
    private static HttpEntity<?> buildHttpEntity(Class<?> sendDataType, HttpMethod method, Map<String, Object> requestHeaders, Map<String, Object> requestParams, String requestBody, Map<String, Object> params, HttpHeaders headers) {
        //1.设置请求参数
        if (requestParams != null && !requestParams.isEmpty()) {
            Iterator<Map.Entry<String, Object>> paramsIterator = requestParams.entrySet().iterator();
            Map.Entry<String, Object> nextParam = null;
            while (paramsIterator.hasNext()) {
                nextParam = paramsIterator.next();
                params.put(nextParam.getKey(), nextParam.getValue());
            }
        }
        //2.设置请求头
        if (requestHeaders != null && !requestHeaders.isEmpty()) {
            Iterator<Map.Entry<String, Object>> headersIterator = requestHeaders.entrySet().iterator();
            Map.Entry<String, Object> nextHeader = null;
            while (headersIterator.hasNext()) {
                nextHeader = headersIterator.next();
                headers.add(nextHeader.getKey(), String.valueOf(nextHeader.getValue()));
            }
        }
        //3.设置请求体
        //4.请求体 请求头封装到HttpEntity
        String contentType = String.valueOf(requestHeaders.get(EtlHttpRequestConstant.CONTENT_TYPE));
        if (requestBody != null && !requestBody.isEmpty()) {
            if (sendDataType == null || sendDataType.equals(JSONObject.class)) {
                Map<String, Object> requestBodyMap = JSONObject.parseObject(requestBody, Map.class);
                Map bodyMap;
                if (MediaType.APPLICATION_JSON_VALUE.equals(contentType) || HttpMethod.GET.equals(method)) {
                    bodyMap = setEntityBody(Map.class, requestBodyMap);
                } else {
                    bodyMap = setEntityBody(LinkedMultiValueMap.class, requestBodyMap);
                }
                return new HttpEntity<>(bodyMap, headers);
            } else if (sendDataType.equals(JSONArray.class)) {
                List<Map> requestBodyMapList = JSON.parseArray(requestBody, Map.class);
                List<Map> bodyMapList = new ArrayList<>();
                if (MediaType.APPLICATION_JSON_VALUE.equals(contentType) || HttpMethod.GET.equals(method)) {
                    requestBodyMapList.forEach(i -> bodyMapList.add(setEntityBody(Map.class, i)));
                } else {
                    requestBodyMapList.forEach(i -> bodyMapList.add(setEntityBody(LinkedMultiValueMap.class, i)));
                }
                return new HttpEntity<>(bodyMapList, headers);
            } else if (sendDataType.equals(String.class)) {
                return new HttpEntity<>(requestBody, headers);
            }

        }
        // todo 如果是其他情况,则返回null
        return new HttpEntity<>(null, headers);
    }

    /**
     * 请求http请求
     *
     * @param restTemplate
     * @param typeClass
     * @param url
     * @param method
     * @param params
     * @param entity
     * @param <T>
     * @return
     */
    private static <T> T handleHttpRequest(RestTemplate restTemplate, Class<T> typeClass, String url, HttpMethod method, Map<String, Object> params, HttpEntity<?> entity) {
        ResponseEntity<T> exchange = restTemplate.exchange(url, method, entity, typeClass, params);

        return typeClass.cast(exchange.getBody());
    }

    /**
     * 处理http请求(字节流)
     *
     * @param restTemplate
     * @param url
     * @param method
     * @param response
     * @param isRespBodyOnly
     * @param params
     * @param entity
     */
    private static void handleHttpRequest(RestTemplate restTemplate, String url, HttpMethod method, HttpServletResponse response, Boolean isRespBodyOnly, Map<String, Object> params, HttpEntity<?> entity) {
        BufferedOutputStream bos = null;
        try {
            ResponseEntity<byte[]> exchange = restTemplate.exchange(url, method, entity, byte[].class, params);
            if (!isRespBodyOnly && response != null) {
                setResponseDetails(response, exchange);
            }
            // FIXME: 2023/3/29 by.xurenyi 修改下载文件流的格式,此处有些小问题,暂定使用原接口的文件格式~~~
//            response.reset();
//            response.setContentType("pdf/plain");
//            response.setContentType("application/octet-stream;charset=UTF-8");
//            response.setHeader("Content-Type", "application/octet-stream;charset=UTF-8");
//            response.setHeader("Content-Disposition", "attachment; filename=" + "test" + ".pdf");
            //响应体
            ServletOutputStream os = response.getOutputStream();
            bos = new BufferedOutputStream(os);
            byte[] buf = exchange.getBody();
            bos.write(buf);
            bos.flush();
        } catch (IOException e) {
            logger.error("RestTemplateUtils获取接口二进制流异常");
        } catch (RestClientException e) {
            logger.error("远程调用接口异常{}", e);
        } finally {
            try {
                bos.close();
            } catch (IOException e) {
                logger.error("RestTemplateUtils流关闭异常");
            }
        }
    }

    /**
     * 设置http响应详情
     *
     * @param response
     * @param exchange
     * @param <T>
     */
    private static <T> void setResponseDetails(HttpServletResponse response, ResponseEntity<T> exchange) {
        //响应状态码
        response.setStatus(exchange.getStatusCodeValue());
        //响应头,可能存在一个key对应多个value,本方法中会将同名header合并
        HttpHeaders resHeaders = exchange.getHeaders();
        Iterator<Entry<String, List<String>>> resHeadersIterator = resHeaders.entrySet().iterator();
        while (resHeadersIterator.hasNext()) {
            Entry<String, List<String>> headersNext = resHeadersIterator.next();
            response.setHeader(headersNext.getKey(), headersNext.getValue().toString());
        }
    }

    /**
     * 设置entity
     *
     * @param beanClass
     * @param requestBody
     * @return
     */
    private static Map setEntityBody(Class<? extends Map> beanClass, Map<String, Object> requestBody) {
        Object entityBody;
        Iterator<Entry<String, Object>> bodyIterator = requestBody.entrySet().iterator();
        Entry<String, Object> bodyNext;

        if (beanClass.equals(MultiValueMap.class) || beanClass.equals(LinkedMultiValueMap.class)) {
            MultiValueMap<String, Object> map = new LinkedMultiValueMap<>();
            while (bodyIterator.hasNext()) {
                bodyNext = bodyIterator.next();
                map.add(bodyNext.getKey(), bodyNext.getValue());
            }

            entityBody = map;
        } else if (beanClass.equals(Map.class)) {
            Map<String, Object> map = new HashMap<>();
            while (bodyIterator.hasNext()) {
                bodyNext = bodyIterator.next();
                map.put(bodyNext.getKey(), bodyNext.getValue());
            }

            entityBody = map;
        } else {
            Map<String, Object> map = new HashMap<>();
            while (bodyIterator.hasNext()) {
                bodyNext = bodyIterator.next();
                map.put(bodyNext.getKey(), bodyNext.getValue());
            }

            entityBody = map;
        }

        return beanClass.cast(entityBody);
    }

    /**
     * RestTemplate 单例 懒汉 双检锁
     **/
    public static RestTemplate getSingleRestTemplate() {
        if (restTemplate == null) {
            synchronized (RestTemplateUtils.class) {
                if (restTemplate == null) {
                    restTemplate = new RestTemplate();
                }
            }
        }
        return restTemplate;
    }

}