Android中URLEncoder空格被轉碼為"+"號的處理辦法
Android前段和后端接口進行交互時,經常會遇到特殊字符,比如表情、特殊標點等,這樣在Url中是無法識別的,需要進行轉碼,后端進行解碼交互。
但當使用URLEncoder時,會發(fā)現字符串中的空格被轉換成“+”號,如果編碼后的內容入庫后,將導致讀取時,前段本來是空格的地方,會顯示成“+”號字符。

到底為什么會這樣,我們還是看看源碼的實現。
首選,URLEncoder中定義了哪些字符不可以被轉碼。
static BitSet dontNeedEncoding;
dontNeedEncoding中就包含了不允許轉碼的字符集合,只有在這里面能找不到的字符才會被正常轉碼。下面是dontNeedEncoding初始化過程
static {
//創(chuàng)建一個容量為256的位集合
dontNeedEncoding = new BitSet(256);
int i;
//循環(huán)a-z,將其字符值(不是字符串值)存入
for (i = 'a'; i <= 'z'; i++) {
dontNeedEncoding.set(i);
}
//同上
for (i = 'A'; i <= 'Z'; i++) {
dontNeedEncoding.set(i);
}
//同上
for (i = '0'; i <= '9'; i++) {
dontNeedEncoding.set(i);
}
//在這個地方,官方特意做了說明
//空格將被編碼為加號
dontNeedEncoding.set(' '); /* encoding a space to a + is done
* in the encode() method */
dontNeedEncoding.set('-');
dontNeedEncoding.set('_');
dontNeedEncoding.set('.');
dontNeedEncoding.set('*');
//默認編碼格式
dfltEncName = AccessController.doPrivileged(
new GetPropertyAction("file.encoding")
);
}上面的代碼我們看到,除了英文字母,數字和幾個能識別的字符外,都需要轉碼。
我們再來看看encode方法。
public static String encode(String s, String enc)
throws UnsupportedEncodingException {
//標記是否有字符被轉碼了,如果沒有,就直接返回原始字符串
//如果有,就返回out的結果
boolean needToChange = false;
//存放轉碼后的結果
StringBuffer out = new StringBuffer(s.length());
Charset charset;
CharArrayWriter charArrayWriter = new CharArrayWriter();
//如果未指定編碼,拋出異常
if (enc == null)
throw new NullPointerException("charsetName");
try {
charset = Charset.forName(enc);
} catch (IllegalCharsetNameException e) {
throw new UnsupportedEncodingException(enc);
} catch (UnsupportedCharsetException e) {
throw new UnsupportedEncodingException(enc);
}
//遍歷要編碼的字符串
for (int i = 0; i < s.length();) {
//取出指定位置的字符
int c = (int) s.charAt(i);
if (dontNeedEncoding.get(c)) {
//這里,判斷如果字符是空格,就轉為“+”
if (c == ' ') {
c = '+';
needToChange = true;
}
out.append((char)c);
i++;
} else {
// convert to external encoding before hex conversion
do {
charArrayWriter.write(c);
/*
* 下面就是轉碼unicode相關代碼
*/
if (c >= 0xD800 && c <= 0xDBFF) {
......
}
i++;
} while (i < s.length() && !dontNeedEncoding.get((c = (int) s.charAt(i))));
charArrayWriter.flush();
String str = new String(charArrayWriter.toCharArray());
byte[] ba = str.getBytes(charset);
for (int j = 0; j < ba.length; j++) {
//添加百分號,比如a'a轉為a%27a
out.append('%');
......
}
charArrayWriter.reset();
needToChange = true;
}
}
return (needToChange? out.toString() : s);
}我們討論的問題,就是下面這幾行。

那么如何處理該問題呢?
注意:網上那種替換空格或加號的方法是不可行的,因為可能會替換掉正常的空格和加號。
建議自己仿照官方的代碼,寫一套自己轉碼和解碼過程,這個不難的?;蛘哒业谌降模ū热鐂pring提供的 UriUtils )。
另外,Android提供了Uri.encode,雖然可以正常轉碼空格,但是它不需要轉換的字符和URLEncode有一點差別,大家在用的時候,根據情況選擇即可。
下面是Uri.encode中不會被轉碼的字符
/**
* Returns true if the given character is allowed.
*
* @param c character to check
* @param allow characters to allow
* @return true if the character is allowed or false if it should be
* encoded
*/
private static boolean isAllowed(char c, String allow) {
return (c >= 'A' && c <= 'Z')
|| (c >= 'a' && c <= 'z')
|| (c >= '0' && c <= '9')
|| "_-!.~'()*".indexOf(c) != NOT_FOUND
|| (allow != null && allow.indexOf(c) != NOT_FOUND);
}總結
到此這篇關于Android中URLEncoder空格被轉碼為"+"號的處理辦法的文章就介紹到這了,更多相關Android URLEncoder空格被轉碼+號內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
android ListView結合xutils3仿微信實現下拉加載更多
本篇文章主要介紹了android ListView結合xutils3仿微信實現下拉加載更多,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-11-11
史上最全Android build.gradle配置詳解(小結)
這篇文章主要介紹了史上最全Android build.gradle配置詳解(小結),小編覺得挺不錯的,現在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2019-04-04
Android入門教程之RecyclerView的具體使用詳解
RecyclerView是Android一個更強大的控件,其不僅可以實現和ListView同樣的效果,還有優(yōu)化了ListView中的各種不足。其可以實現數據縱向滾動,也可以實現橫向滾動(ListView做不到橫向滾動)。接下來講解RecyclerView的用法2021-10-10
Android使用RecyclerView實現列表數據選擇操作
這篇文章主要為大家詳細介紹了Android使用RecyclerView結合CheckBox實現列表數據選擇操作,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-06-06
Android?RecyclerBarChart繪制使用教程
這篇文章主要為大家介紹了Android?RecyclerBarChart繪制使用教程,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-12-12

