Android中Json字符串解析总结

最近的项目里面对解析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就能满足你的要求了

你可能感兴趣的:(Android中Json字符串解析总结)