欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

基于restTemplate遇到的編碼問題及解決

 更新時(shí)間:2021年10月28日 10:44:28   作者:zhuxingKevin  
這篇文章主要介紹了restTemplate遇到的編碼問題及解決方案,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教

背景

之前用restTemplate做網(wǎng)絡(luò)間的請求,沒遇到過問題。今天先是出現(xiàn)了中文亂碼的問題,而后又出現(xiàn)了特殊字符丟失的問題,于是查找資料及翻看源碼,將問題解決也順便記錄下。

問題一:中文亂碼

描述

在創(chuàng)建課件時(shí),使用GET方法傳遞類型和標(biāo)題兩個(gè)參數(shù)到服務(wù)器,服務(wù)器返回一個(gè)課件編號。類型是固定數(shù)字1,不存在問題,而標(biāo)題則是用戶輸入字符串,也就是任意字符串。發(fā)現(xiàn)輸入漢字的時(shí)候,結(jié)果網(wǎng)絡(luò)傳輸后在服務(wù)器端出現(xiàn)了亂碼。輸入標(biāo)題為:開發(fā)測試001,結(jié)果在服務(wù)器上接收到的為:%E5%BC%80%E5%8F%91%E6%B5%8B%E8%AF%95001。

分析

出現(xiàn)編碼問題,那肯定是對涉及到這個(gè)標(biāo)題的編碼的位置出現(xiàn)了問題,于是查找涉及到的代碼位置:

public String createPptSlide(String title) {
        UriComponentsBuilder builder = UriComponentsBuilder.fromHttpUrl(host + PPT_SLIDE_INSERT);
        builder.queryParam("businessLineId", PPT_BUSINESS_LINE_ID);
        builder.queryParam("slideTitle", title);
        String url = builder.toUriString();
        restTemplate.getForEntity(url, JSONObject.class);
}

此處的String url已經(jīng)是編碼過了的http://xxx.com/slide/insertEmptySlide?businessLineId=1&slideTitle=%E5%BC%80%E5%8F%91%E6%B5%8B%E8%AF%95001

于是繼續(xù)追蹤getForEntity方法,在執(zhí)行doExecute方法前,構(gòu)造URI時(shí)又進(jìn)行了一次編碼,如下:

@Override
    public <T> ResponseEntity<T> getForEntity(String url, Class<T> responseType, Map<String, ?> urlVariables)
            throws RestClientException {
        RequestCallback requestCallback = acceptHeaderRequestCallback(responseType);
        ResponseExtractor<ResponseEntity<T>> responseExtractor = responseEntityExtractor(responseType);
        return execute(url, HttpMethod.GET, requestCallback, responseExtractor, urlVariables);
    }
public <T> T execute(String url, HttpMethod method, RequestCallback requestCallback,
            ResponseExtractor<T> responseExtractor, Map<String, ?> urlVariables) throws RestClientException {
        URI expanded = new UriTemplate(url).expand(urlVariables);
        return doExecute(expanded, method, requestCallback, responseExtractor);
}
public URI expand(Map<String, ?> uriVariables) {
        UriComponents expandedComponents = this.uriComponents.expand(uriVariables);
        UriComponents encodedComponents = expandedComponents.encode();
        return encodedComponents.toUri();
}

結(jié)論

在程序構(gòu)建url時(shí),程序代碼已經(jīng)對title進(jìn)行了一次編碼,,傳入restTemplate后,restTemplate框架本身又會(huì)對其做一次編碼,最后服務(wù)器接收到的參數(shù)其實(shí)是做了兩次編碼,導(dǎo)致解碼后還是亂碼。

第一次編碼:%E5%BC%80%E5%8F%91%E6%B5%8B%E8%AF%95001

第二次編碼:%25E5%25BC%2580%25E5%258F%2591%25E6%25B5%258B%25E8%25AF%2595001

解碼后:%E5%BC%80%E5%8F%91%E6%B5%8B%E8%AF%95001

側(cè)面論證:

這里的編碼是URLencode,這個(gè)編碼簡單來講就是:將非字母數(shù)字字符都將被替換成百分號(%)后跟兩位十六進(jìn)制數(shù)??梢钥吹竭@一串的亂碼%E5%BC%80%E5%8F%91%E6%B5%8B%E8%AF%95001很像urlencode后的結(jié)果,解碼后發(fā)現(xiàn)果真是這樣。

image.png

方案

傳入url前不對title進(jìn)行編碼,直接拼接原始的參數(shù),然后傳入到restTemplate,讓restTemplate框架來進(jìn)行編碼。這樣解決了中文亂碼的問題,如下:

UriComponentsBuilder builder = UriComponentsBuilder.fromHttpUrl(host + PPT_SLIDE_INSERT);
        builder.queryParam("businessLineId", PPT_BUSINESS_LINE_ID);
        String url = builder.toUriString();
        // 不能 encode 參數(shù)
        if (StringUtils.isNotBlank(title)) {
            url = url + "&slideTitle=" + title;
        }
        restTemplate.getForEntity(url, JSONObject.class);
}

總結(jié)

項(xiàng)目中使用restTemplate的地方,在傳給restTemplate框架url的時(shí)候都進(jìn)行了一次編碼,于是自己也照搬過來,殊不知restTemplate框架本身就會(huì)對url進(jìn)行編碼。

其實(shí)從一個(gè)框架設(shè)計(jì)者的角度上將,urlencode是每個(gè)請求都會(huì)用到的,很順利的可以想到框架會(huì)包含這個(gè)urlencode功能,不需要編程人員將參數(shù)urlencode。

問題二:特殊字符串丟失

描述

在使用restTemplate傳遞課件標(biāo)題的時(shí)候,發(fā)現(xiàn)有一些特殊字符串有數(shù)據(jù)丟失問題,例如傳遞課件標(biāo)題:開發(fā)測試001&aaa,接收方接到的是:開發(fā)測試001,后面跟的&aaa字符丟失了。

分析

再一次懷疑是restTemplate里面自帶的編碼導(dǎo)致的問題,于是對比了直接使用urlencode與使用restTemplate編碼的結(jié)果,如下表:

urlencode

%e5%bc%80%e5%8f%91%e6%b5%8b%e8%af%95001%26aaa

restTemplate

%E5%BC%80%E5%8F%91%E6%B5%8B%E8%AF%95001&aaa

明顯可以發(fā)現(xiàn)restTemplate對特殊字符“&”沒有進(jìn)行url編碼,導(dǎo)致最后構(gòu)建成的url是這樣的:

http://xxx.com/slide/insertEmptySlide?businessLineId=1&slideTitle=%E5%BC%80%E5%8F%91%E6%B5%8B%E8%AF%95001&aaa

這樣就會(huì)把&aaa中的aaa當(dāng)成get請求中的一個(gè)參數(shù)來看待,從而得到的title只為:%E5%BC%80%E5%8F%91%E6%B5%8B%E8%AF%95001,導(dǎo)致丟失了&aaa字符,接收方解析也只能得到標(biāo)題為:開發(fā)測試001。

結(jié)論

restTemplate在進(jìn)行url編碼的時(shí)候,不會(huì)對某些特殊字符的編碼,例如&。

方案

既然restTemplate會(huì)忽略掉某些特殊字符的url編碼,那么我們就索性不用restTemplate編碼,直接自己編碼好,跳過restTemplate的編碼,實(shí)現(xiàn)方案為:

public String createPptSlide(String title) {
        UriComponentsBuilder builder = UriComponentsBuilder.fromHttpUrl(host + PPT_SLIDE_INSERT);
        builder.queryParam("businessLineId", PPT_BUSINESS_LINE_ID);
        builder.queryParam("slideTitle", title);
        URI uri = builder.build().encode().toUri();
        restTemplate.getForEntity(uri, JSONObject.class);
}

(ps:傳String的url,restTemplate都會(huì)再一次進(jìn)行編碼,而直接傳URI可以跳過restTemplate的編碼)

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

相關(guān)文章

  • 關(guān)于Mybatis實(shí)體別名支持通配符掃描問題小結(jié)

    關(guān)于Mybatis實(shí)體別名支持通配符掃描問題小結(jié)

    MyBatis可以使用簡單的 XML 或注解來配置和映射原生信息,將接口和 Java 的 POJOs(Plain Old Java Objects,普通的 Java對象)映射成數(shù)據(jù)庫中的記錄,這篇文章主要介紹了Mybatis實(shí)體別名支持通配符掃描的問題,需要的朋友可以參考下
    2022-01-01
  • String轉(zhuǎn)JSONObject的兩種方式

    String轉(zhuǎn)JSONObject的兩種方式

    這篇文章主要介紹了String轉(zhuǎn)JSONObject,本文通過實(shí)例代碼給大家介紹兩種方式轉(zhuǎn)換,結(jié)合實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2023-07-07
  • RabbitMQ?延遲隊(duì)列實(shí)現(xiàn)訂單支付結(jié)果異步階梯性通知(實(shí)例代碼)

    RabbitMQ?延遲隊(duì)列實(shí)現(xiàn)訂單支付結(jié)果異步階梯性通知(實(shí)例代碼)

    這篇文章主要介紹了RabbitMQ?延遲隊(duì)列實(shí)現(xiàn)訂單支付結(jié)果異步階梯性通知,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2022-02-02
  • SpringBoot+Vue+Element-ui實(shí)現(xiàn)前后端分離

    SpringBoot+Vue+Element-ui實(shí)現(xiàn)前后端分離

    使用前后端分離的方式,可以減少代碼耦合,本文主要介紹了SpringBoot+Vue+Element-ui實(shí)現(xiàn)前后端分離,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-06-06
  • java 內(nèi)部類的實(shí)例詳解

    java 內(nèi)部類的實(shí)例詳解

    這篇文章主要介紹了java 內(nèi)部類的實(shí)例詳解的相關(guān)資料,希望通過本文大家能夠理解掌握java內(nèi)部類的使用,需要的朋友可以參考下
    2017-09-09
  • java中的抽象類和接口定義與用法詳解

    java中的抽象類和接口定義與用法詳解

    在面向?qū)ο蟮母拍钪?,所有的對象都是通過類來描繪的,但是反過來,并不是所有的類都是用來描繪對象的,如果一個(gè)類中沒有包含足夠的信息來描繪一個(gè)具體的對象,這樣的類就是抽象類
    2021-10-10
  • IntelliJ IDEA多屏后窗口不顯示問題解決方案

    IntelliJ IDEA多屏后窗口不顯示問題解決方案

    這篇文章主要介紹了IntelliJ IDEA多屏后窗口不顯示問題解決方案,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-09-09
  • Spring學(xué)習(xí)筆記之RestTemplate使用小結(jié)

    Spring學(xué)習(xí)筆記之RestTemplate使用小結(jié)

    這篇文章主要給大家介紹了關(guān)于Spring學(xué)習(xí)筆記之RestTemplate使用的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2018-08-08
  • SpringCloud Zuul過濾器和谷歌Gauva實(shí)現(xiàn)限流

    SpringCloud Zuul過濾器和谷歌Gauva實(shí)現(xiàn)限流

    這篇文章主要介紹了SpringCloud Zuul過濾器和谷歌Gauva實(shí)現(xiàn)限流,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-03-03
  • spring security數(shù)據(jù)庫表結(jié)構(gòu)實(shí)例代碼

    spring security數(shù)據(jù)庫表結(jié)構(gòu)實(shí)例代碼

    這篇文章主要介紹了spring security數(shù)據(jù)庫表結(jié)構(gòu)實(shí)例代碼,需要的朋友可以參考下
    2017-09-09

最新評論