最近的项目里面对解析Json出现了一些问题,主要集中在iOS和Android两者在同一个字符串识别的时候出现的结果不同
先看一个Json例子
{
"aa": 1,
"bb": "bb",
"cc": true
}
这是一个简单的Json字符串,包含了数字,字符串和bool值,而引起这篇文章的动机就是cc的bool值,在标准Json中对bool值的定义是 true或者false,也就是说只有标准的小写并且不带引号的这两个字符串才是正确的bool值。
类似于True和False这样不标准的bool值在标准Json中是错误的,但是强大的抓包工具能够让我们对发送的请求和response的数据进行更改,也就是说在中途如果我们将标准的true改成True,那么会发生什么呢?接下来请看下面分解。
为了和上面的Json数据对应,我们创建一个对象
data class JsonData(var aa: Int, var bb: String, var cc: Boolean)
Google的Gson库进行解析
val jsonStr = "{\"aa\": 1,\"bb\": \"bb\",\"cc\": true}"
@Test
fun testParseJsonByGson(){
val jsonObj = Gson().fromJson(jsonStr, JsonData::class.java)
assertEquals(jsonObj.cc, true)
}
上面的代码直接将字符串转换成了对象,没有问题,下面我们用同样的方法将Json字符串里面的true改成大写的True试试
val jsonStr = "{\"aa\": 1,\"bb\": \"bb\",\"cc\": True}"
@Test
fun testParseJsonByGson(){
val jsonObj = Gson().fromJson(jsonStr, JsonData::class.java)
assertEquals(jsonObj.cc, true)
}
这样测试也没问题,这就表示Gson库是大小写不敏感的,好了,现在我们来看看Gson关于这部分的源码就自知道了,找到 com.google.gson.stream.JsonReader
类,里面的peekKeyword方法,如下
if (c == 't' || c == 'T') {
keyword = "true";
keywordUpper = "TRUE";
peeking = PEEKED_TRUE;
} else if (c == 'f' || c == 'F') {
keyword = "false";
keywordUpper = "FALSE";
peeking = PEEKED_FALSE;
} else if (c == 'n' || c == 'N') {
keyword = "null";
keywordUpper = "NULL";
peeking = PEEKED_NULL;
} else {
return PEEKED_NONE;
}
可以看出Gson里面是部分大小写的,不管是T开头还是t开头的true都会认为是bool类型。
而如果我们将JsonData类中的cc改成String类型的呢?
data class JsonData(var aa: Int, var bb: String, var cc: String)
这时你会发现Gson会将bool中的True转换成true,然后再把小写的true转换成字符串的 true
。而如果此时是Tru
这样一个错误的值,那么就不会被转换成小写的true然后再转换成字符串,而是直接转换成字符串 "Tru"这样大写的。
阿里的FastJson进行解析
这里不重复说明,在正确的书写情况下,FastJson和Gson是一样的解析结果,而如果是下面这种情况,将 True
进行大写
val jsonStr = "{\"aa\": 1,\"bb\": \"bb\",\"cc\": True}"
再使用FastJson进行解析
@Test
fun testParseJsonByFastJson(){
val jsonObj = JSON.parseObject(jsonStr, JsonData::class.java)
assertEquals(jsonObj.cc, true)
}
结果是抛出异常
om.alibaba.fastjson.JSONException: default constructor not found.
我们去FastJson的源码看看为什么会这样,找到 com.alibaba.fastjson.parser.DefaultJSONParser中的parseObject方法
if (lexer.text.startsWith("true", lexer.bp)) {
lexer.bp += 3;
lexer.next();
object.put(key, Boolean.TRUE);
}
} else if (ch == 'f') {
if (lexer.text.startsWith("false", lexer.bp)) {
lexer.bp += 4;
lexer.next();
object.put(key, Boolean.FALSE);
}
这段代码对true和false进行了限制,只有当这个值符合小写的true和false时才被认为是bool值,如果不是,我们这里不关心,只要明白如果不是小写的字母那么就不被认可为bool值了。
总结
这次的问题出在之前对Gson解析的不完全了解,而Gson库对出现的Json对象数据进行了最大程度的包容处理,这里不能说谁对错,关键看项目怎么用了,如果要求严格一点,那么就换成FastJson这样的三方库来保证,如果要求更松那么Gson就能满足你的要求了