现在开源的 Json 序列化方案层出不穷,在性能和使用方面,都取得了很好的效果,比较常用的包括 Gson,FastJson 等等。然后对于初学者而言,或者不想引入额外框架的情况下,在这些场景下,还是会使用 JsonObject 这种基础对象。今天的文章,就是说一说 JsonObject 的神坑。

来看看这个例子,这样的一个 Json 键值对。

1
2
3
{
"userName":null
}

和下面的这个 Json 键值对。

1
2
3
{
"userName":"null"
}

区别就在 value 上面,一个是实际的 null,另一个是字符串 “null”。我们知道这两个区别很大,但本着科学的态度,我们先用 Gson 来做一下测试,看看情况是否是我们想象中的那样。

新建了一个 People 的类,方便通过 Gson 进行反序列化。

1
2
3
4
5
6
7
public class People implements Serializable {
private static final long serialVersionUID = -7182026066461954852L;
public String userName;
}

再通过实际的代码,来进行测试。

1
2
3
4
5
6
7
8
9
Gson gson = new Gson();
People p1= gson.fromJson("{\n" +
" \"userName\":null\n" +
"}", People.class);
People p2 = gson.fromJson("{\n" +
" \"userName\":\"null\"\n" +
"}", People.class);
Log.d("tango", String.valueOf(TextUtils.isEmpty(p1.userName)));
Log.d("tango", String.valueOf(TextUtils.isEmpty(p2.userName)));

最后输出的结果,分别是 truefalse, 符合我们的预期。

接下来看看 JsonObject 的表现如何,代码很简单,如下所示。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
try {
JSONObject jsonObject1 = new JSONObject("{\n" +
" \"userName\":null\n" +
"}");
Log.d("tango", jsonObject1.getString("userName"));
} catch (JSONException e) {
e.printStackTrace();
}
try {
JSONObject jsonObject2 = new JSONObject("{\n" +
" \"userName\":\"null\"\n" +
"}");
Log.d("tango", jsonObject2.getString("userName"));
} catch (JSONException e) {
e.printStackTrace();
}

这个地方不会崩溃吗?按道理上讲 jsonObject1.getString("userName") 这里是 null 呀,可是没有!它真的没有,真的没有。即使是 null,不是字符串null,JsonObject 也能神奇地返回字符串 null, 这真是天坑呀,比地缝都大啊。

天啊
天啊

在知道原因以后,解决起来也很容易,在 JsonObject 提供了判断的方法,是 JsonObject.isNull。当然,可以的话,尽量不用 JsonObject ,毕竟不但慢,而且代码可读性差。


文档信息