亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

SpringBoot之RestTemplate在URL中轉(zhuǎn)義字符的問題

 更新時(shí)間:2023年06月26日 08:57:33   作者:MLn0829  
這篇文章主要介紹了SpringBoot之RestTemplate在URL中轉(zhuǎn)義字符的問題,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教

RestTemplate在URL中轉(zhuǎn)義字符的問題

相同代碼在不同SpringBoot版本中使用RestTemplate請求時(shí)遇到的問題:

Map<String, Object> param = new HashMap<String, Object>();
            param.put("version","2.0.0");
            String userSing = sign(param);
            param.put("user_sign", userSing);
            logger.info("簽名:" + userSing);
            StringBuilder paramStr = new StringBuilder("?");
            for(Map.Entry<String, Object> entry : param.entrySet()){
                paramStr.append(entry.getKey()).append("=")
                        .append(entry.getValue() == null ? "" : String.valueOf(entry.getValue()))
                        .append("&");
            }
            paramStr.deleteCharAt(paramStr.length() - 1);
            logger.info("入?yún)?" + paramStr.toString());
            RestTemplate restTemplate = new RestTemplate();
            String jsonStr = restTemplate.getForObject(sendMessagesUrl + paramStr.toString(), String.class);
            logger.info("響應(yīng)值:" + jsonStr);

下面截圖為springboot1.5.3 RestTemplate request log

springboot1.5.3 RestTemplate request log

下面截圖為springboot2.1.7 RestTemplate request log

springboot2.1.7 RestTemplate request log

在這兩份log中可以看到user_sign對應(yīng)的value值中的 + 是特殊的字符,1.0版本的被轉(zhuǎn)義為: %2B2.0版本沒有被轉(zhuǎn)義,最終2.0版本程序的RestTemplate請求第三方在簽名解碼時(shí)校驗(yàn)不通過。

1.嘗試與分析

根據(jù)上述信息我們可以圈定問題的范圍,= 有被轉(zhuǎn)碼,說明可能是在RestTemplate中url參數(shù)的構(gòu)建轉(zhuǎn)碼的方式上與httpClient有什么不通,嘗試進(jìn)行各類處理方法。

主要使用的方式有:

  • UriComponent構(gòu)建uri(未解決)
  • 構(gòu)建如下RestTemplateConfig(未解決)
public class RestTemplateConfig {
    @Bean
    RestTemplate restTemplate() {
        SimpleClientHttpRequestFactory requestFactory = new SimpleClientHttpRequestFactory();
        requestFactory.setReadTimeout(40000);
        requestFactory.setConnectTimeout(40000);
        // 添加轉(zhuǎn)換器
        List<HttpMessageConverter<?>> messageConverters = new ArrayList<>();
        messageConverters.add(new StringHttpMessageConverter(Charset.forName("UTF-8")));
        messageConverters.add(new FormHttpMessageConverter());
        messageConverters.add(new MappingJackson2HttpMessageConverter());
        RestTemplate restTemplate = new RestTemplate(messageConverters);
        restTemplate.setRequestFactory(requestFactory);
        restTemplate.setErrorHandler(new DefaultResponseErrorHandler());
        return restTemplate;
    }
}

2.跟蹤RestTemplate源碼

RestTemplate源碼

debug到這里時(shí)發(fā)現(xiàn)user_sign被轉(zhuǎn)義了但 + 沒有變。

URLDecode

這時(shí)URLDecoder.decode("/QaNSBls/U8ciXEWaCWtmeMK6+w%3D")會發(fā)現(xiàn) + 變成了 空格,導(dǎo)致第三方在簽名解碼時(shí)校驗(yàn)不通過。

isAllowed

再次通過比較會發(fā)現(xiàn)StringBoot2.0版本對應(yīng)的URL轉(zhuǎn)碼 + 不會被解析。

經(jīng)過多次百度有看到如下:

根據(jù) RFC 3986 加號等符號的確實(shí)可以出現(xiàn)在參數(shù)中的,而且不需要編碼,有問題的在于服務(wù)端的解析沒有與時(shí)俱進(jìn)

3.解決問題

最后解決一些這個(gè)問題,當(dāng)使用RestTemplate發(fā)起請求,如果請求參數(shù)中有需要url編碼時(shí)

通過如下兩種方式解決:

1.不希望出現(xiàn)問題的使用姿勢應(yīng)傳入U(xiǎn)RI對象而不是字符串,修改RestTemplate請求方法入?yún)⒌?code>String url 修改為 URI url。

2.修改入?yún)⒕幋a格式URLEncoder.encode(user_sign, "UTF-8"),然后在構(gòu)建RestTemplate時(shí),

Map<String, Object> param = new HashMap<String, Object>();
        StringBuilder paramStr = new StringBuilder("?");
            param.put("version","xxxxx");
            param.put("time","xxxxx");
            //轉(zhuǎn)化成md5生產(chǎn)密鑰
            String userSing  = sign(param);
            String userSign = URLEncoder.encode(userSing, "UTF-8");
            param.put("user_sign", userSign);
            for(Map.Entry<String, Object> entry : param.entrySet()){
                paramStr.append(entry.getKey()).append("=")
                        .append(entry.getValue() == null ? "" : String.valueOf(entry.getValue()))
                        .append("&");
            }
            paramStr.deleteCharAt(paramStr.length() - 1);
            CloseableHttpClient httpClient = HttpClientUtils.acceptsUntrustedCertsHttpClient();
            HttpComponentsClientHttpRequestFactory clientHttpRequestFactory = new HttpComponentsClientHttpRequestFactory(httpClient);
            RestTemplate restTemplate = new RestTemplate(clientHttpRequestFactory);
            DefaultUriBuilderFactory uriFactory = new DefaultUriBuilderFactory();
            uriFactory.setEncodingMode(DefaultUriBuilderFactory.EncodingMode.VALUES_ONLY);
            restTemplate.setUriTemplateHandler(uriFactory);
            String smsJsonStr = restTemplate.getForObject(SMSURL + paramStr.toString(), String.class);
            Map<String, Object> map = GsonUtil.gsonToMap(smsJsonStr);

小結(jié)

注意SpringBoot2.0版本的url參數(shù)編碼,默認(rèn)只會針對 = 和 & 進(jìn)行處理;為了兼容我們一般的后端的url編解碼處理在需要編碼參數(shù)時(shí),

個(gè)人建議盡量不要使用Spring默認(rèn)的方式,不然接收到數(shù)據(jù)會和預(yù)期的不一致。

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。

相關(guān)文章

最新評論