我试图反序列化以下结构
{ meta: { keywords: [a, b, c, d]} ... }
其他有效结构
{ meta: { keywords: "a,b,c,d"} ... }
和
{ meta: {keywords: "a"} ...}
我有这个课程
public class Data {
@PropertyName("meta")
MetaData meta;
...
}
public class MetaData {
List<String> keywords;
...
}
和自定义反序列化程序
public static class CustomDeserilizer implements JsonDeserializer<MetaData>{
@Override
public MetaData deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) throws JsonParseException {
List<String> keywords = null;
Gson gson = new Gson();
MetaData metaData = gson.fromJson(json, AppMetaData.class);
JsonObject jsonObject = json.getAsJsonObject();
if (jsonObject.has("keywords")) {
JsonElement elem = jsonObject.get("keywords");
if (elem != null && !elem.isJsonNull()) {
if (jsonObject.get("keywords").isJsonArray()) {
keywords = gson.fromJson(jsonObject.get("keywords"), new TypeToken<List<String>>() {
}.getType());
} else {
String keywordString = gson.fromJson(jsonObject.get("keywords"), String.class);
keywords = new ArrayList<String>();
list.addAll(Arrays.asList(keywordString.split(",")));
}
}
}
metaData.setKeywords(keywords);
}
然后我尝试应用deserilizer:
Gson gson = new GsonBuilder()
.registerTypeAdapter(Data.class,new CustomDeserilizer())
.create();
但是我得到一个解析错误,因为我试图反序列化Data而不是MetaData,我如何应用这个反序列化器使其正常工作?
答案 0 :(得分:5)
我解决了为我的类Data创建反序列化器的问题。
public static class DataDeserilizer implements JsonDeserializer {
@Override
public Data deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) throws JsonParseException {
Gson gson = new Gson();
Data data = gson.fromJson(json, Data.class);
JsonObject jsonObject = json.getAsJsonObject();
if (jsonObject.has("meta")) {
JsonElement elem = jsonObject.get("meta");
if (elem != null && !elem.isJsonNull()) {
Gson gsonDeserializer = new GsonBuilder()
.registerTypeAdapter(MetaData.class, new CustomDeserilizer())
.create();
gsonDeserializer.fromJson(jsonObject.get("meta"), Data.class);
}
}
return data;
}
}
并且
Gson gson = new GsonBuilder()
.registerTypeAdapter(Data.class,new DataDeserilizer())
.create();
很明显,但有更优雅的解决方案吗?
答案 1 :(得分:0)
首先,将您的类重命名为meta而不是元数据,并使关键字String而不是List.Then使用以下内容将您的JSonString映射到您的对象。
Gson gson = new GsonBuilder().create();
Meta meta = gson.from(yourJsonString,Meta.class);
要获得关键字,您需要这样做。
JsonObject jsonObject = new JsonObject(yourJSonString);
String data = jsonObject.getJsonObject("meta").getString("keywords");
关键字是JsonObject而不是JsonArray,因此您无法直接映射它 到列表。您可以拆分字符串以获取数组中的关键字。
String keywords[] = data.split(",");
答案 2 :(得分:0)
这是一个简洁的解决方案,利用Java继承来表示嵌套结构;因此,不需要提供任何实际的实例成员字段(映射等)来捕获GSON映射的嵌套String
数据。
步骤1:为了便于阅读,请创建一个空对象来表示嵌套映射
public class StateRegionCitiesMap extends HashMap<String, List<String>> {
}
步骤2::添加一行实际代码来进行映射; 没有其他要管理的序列化/反序列化逻辑
protected void loadContent(JsonObject stateRegionsJsonObject) {
HashMap<String, StateRegionCitiesMap> stateRegionCitiesMap =
mGson.fromJson(
stateRegionsJsonObject,
new TypeToken<HashMap<String, StateRegionCitiesMap>>() {
}.getType()
);
}
或者,您可以完全跳过包装器类,而直接将<String, List<String>>
放在GSON调用中。但是,我发现一个显式对象有助于告知/提醒正在阅读代码的人,目的是什么。
示例JSON:
类StateRegionCitiesMap
代表多层地图结构,例如:
[美国州]-> [州区域键]-> [子区域键]-> CitiesArray []
"CA": {
"Central CA": {
"Central Valley": [
"FRESNO",
"VISALIA"
],
"Sacramento Area": [
"SACRAMENTO",
"EL DORADO HILLS"
]
},