笔者前几周在处理数据解析这块碰到了好多的坑
这里只讨论数据解析这块的知识
这里并不打算直接通过object = new Gson().fromJson(json, object.getClass());
得到对应的json的实体类型
而是通过JsonParser Api对数据进行解析,方便后面讲到对于复杂特殊的json串进行解析
通过上图的代码片段可以清晰的看到对于一个json串,使用JsonParser api解析后,它就是一个JsonElement对象了,但是这个JsonElement到底是什么,对应的到底是哪一类呢,看了代码可以知道,总共就四类,JsonNull(空串这样的),JsonObject,JsonArray,JsonPrimitive,相信接触过json的同学对JsonObject,JsonArray很熟悉了吧,用的最多的就是这两个,JsonNull用的少,JsonPrimitive估计都没有听过吧
JsonNull没什么好讲的,自己做实验就知道了,JsonPrimitive表示的是json串里面的内容是基础类型,可以看类的定义
当然了也存在特殊的情况,它也是JsonPrimitive类型
我们再看一段代码片段
定义了一个string变量
String json = "{\"code\":200, \"message\":\"ok\", \"data\":\"{\\\"id\\\":131,\\\"appId\\\":6,\\\"versionCode\\\":6014000}\"}";
我们需要关注的是json串里面key为“data”所对应的value,应该看得比较清楚,data的内容都带有转义符号,那么我们该怎么解析这样的数据呢?
回答问题前,我们看看这段代码的执行结果,最终输出的是
json == {"code":200, "message":"ok", "data":"{\"id\":131,\"appId\":6,\"versionCode\":6014000}"}
isJsonObject
isJsonPrimitive == true
可以看到data的内容就是一个jsonPrimitive类型的,因为它在后端处理的时候,就将真实的内容给拍平了(不知道能不能用序列化描述额,总之是被转义了,看起来就不是一个json串,而变成了一个纯粹的string)
那么在我们解析这个被转义的字符串的时候,需要把转义符号去掉。
JsonPrimitive primitive = jsonElement.getAsJsonObject().get("data").getAsJsonPrimitive();
System.out.println("getAsString == " + primitive.getAsString());
System.out.println("toString == " + primitive.toString());
得到的结果是
getAsString == {"id":131,"appId":6,"versionCode":6014000}
toString == "{\"id\":131,\"appId\":6,\"versionCode\":6014000}"
看这个执行结果可以知道通过getAsString(),得到的就是一个json串,而不是还带有转义字符的string,这种情况下,就可以通过
App app = new Gson().fromJson(primitive.getAsJsonPrimitive().getAsString(), App.class);
得到具体的实体类模型了
以上只是简单的情况,复杂的情况需要结合真实的实际场景来,使用标准的Gson api解析,这样可以避免出现一些莫名其妙的问题
举一个例子来描述这种场景:
这个代码片段试将后端的一个json数据转换成了map的数据结构,我们看下结果
{code=200.0, message=ok, data=dataString}
发现code被转成了200.0,变成了double类型,这个问题在这里不深究,想深究的可以看这个https://gist.github.com/xingstarx/5ddc14ff6ca68ba4097815c90d1c47cc,以及https://stackoverflow.com/a/36529534/5279354 总之在使用Gson解析的时候,不要只知道gson.from这一个api,没事也需要多看看Gson相关的其他api,了解下Gson解析的工作原理,有助于我们更好的处理这类问题
附上完整的代码片段
@Test
public void test2() {
String json = "{\"code\":200, \"message\":\"ok\", \"data\":\"{\\\"id\\\":131,\\\"appId\\\":6,\\\"versionCode\\\":6014000}\"}";
System.out.println("json == " + json);
JsonElement jsonElement = new JsonParser().parse(json);
if (jsonElement.isJsonNull()) {
} else if (jsonElement.isJsonObject()) {
JsonPrimitive primitive = jsonElement.getAsJsonObject().get("data").getAsJsonPrimitive();
System.out.println("getAsString == " + primitive.getAsString());
System.out.println("toString == " + primitive.toString());
App app = new Gson().fromJson(primitive.getAsJsonPrimitive().getAsString(), App.class);
// App app = new Gson().fromJson(primitive.getAsJsonPrimitive().toString(), App.class);
System.out.println(app);
} else if (jsonElement.isJsonArray()) {
} else if (jsonElement.isJsonPrimitive()) {
}
// Map<String, Object> map = new LinkedTreeMap<>();
// map = new Gson().fromJson(json, map.getClass());
// System.out.println(map);
}