遇到一坑,以前用的是Gson庫,現在改為Android自帶的JSON解析,后端收到數據有時會解析失敗,查找原因,發現Gson庫和Android自帶JSON解析對轉義字符的處理存在稍許的不同。
舉例如下:
String s1 = "https://blog.csdn.net/robotech_er/article/details/40260377";
String s2 = "host is \"www.baidu.com\"!";
// Android自帶JSON解析
JSONArray ja = new JSONArray();
ja.put(s1);
ja.put(s2);
Log.d("json", ja.toString());
// Gson庫
JsonArray array = new JsonArray();
array.add(new JsonPrimitive(s1));
array.add(new JsonPrimitive(s2));
Log.d("gson", array.toString());
輸出為:
D/json: ["https:\/\/blog.csdn.net\/robotech_er\/article\/details\/40260377","host is \"www.baidu.com\"!"]
D/gson: ["https://blog.csdn.net/robotech_er/article/details/40260377","host is \"www.baidu.com\"!"]
可發現:
(1)Gson庫和Android自帶JSON解析都對引號(")進行了轉義,轉為了"。
(2)Android自帶JSON解析對“/”進行了轉義,轉為了“/”,但是Gson庫沒這么做。
猜測:Gson只對影響json格式的符號進行了轉義,比如引號,但是沒轉義不影響json格式的符號,比如“/”。
查看代碼驗證==>>
Gson庫源碼地址:https://github.com/google/gson
Gson庫對String的轉義在gson/src/main/java/com/google/gson/stream/JsonWriter.java中
static {
REPLACEMENT_CHARS = new String[128];
for (int i = 0; i <= 0x1f; i++) {
REPLACEMENT_CHARS[i] = String.format("\\u%04x", (int) i);
}
REPLACEMENT_CHARS['"'] = "\\\"";
REPLACEMENT_CHARS['\\'] = "\\\\";
REPLACEMENT_CHARS['\t'] = "\\t";
REPLACEMENT_CHARS['\b'] = "\\b";
REPLACEMENT_CHARS['\n'] = "\\n";
REPLACEMENT_CHARS['\r'] = "\\r";
REPLACEMENT_CHARS['\f'] = "\\f";
HTML_SAFE_REPLACEMENT_CHARS = REPLACEMENT_CHARS.clone();
HTML_SAFE_REPLACEMENT_CHARS['<'] = "\\u003c";
HTML_SAFE_REPLACEMENT_CHARS['>'] = "\\u003e";
HTML_SAFE_REPLACEMENT_CHARS['&'] = "\\u0026";
HTML_SAFE_REPLACEMENT_CHARS['='] = "\\u003d";
HTML_SAFE_REPLACEMENT_CHARS['\''] = "\\u0027";
}
Android自帶的JSON解析對String的轉義實現在org/json/JSONStringer
private void string(String value) {
out.append("\"");
for (int i = 0, length = value.length(); i < length; i++) {
char c = value.charAt(i);
switch (c) {
case '"':
case '\\':
case '/': // Gson庫少了對這個的轉義~~~
out.append('\\').append(c);
break;
case '\t':
out.append("\\t");
break;
case '\b':
out.append("\\b");
break;
case '\n':
out.append("\\n");
break;
case '\r':
out.append("\\r");
break;
case '\f':
out.append("\\f");
break;
default:
if (c <= 0x1F) {
out.append(String.format("\\u%04x", (int) c));
} else {
out.append(c);
}
break;
}
}
out.append("\"");
}
對比一下可以發現,Gson庫對HTML有幾個特殊處理,不太懂HTML所以先忽略,
除此之外,Android自帶的JSON解析就比Gson庫多了對"/"的轉義處理!!
那么官方是怎么規定這個轉義的呢,
http://www.json.org上有張圖片很容易說明
json.org
可以看到,里面是有對“/”轉義的
在看下RFC文檔:
RFC文檔
可以看到,里面也是有對“/”轉義的說明的。可能是Google認為無需對這個“/”進行轉義吧,因為對于json格式來說,沒什么影響。