我有一个JSON格式的树对象我试图用Gson反序列化。每个节点包含其子节点作为对象类型Node的字段。 Node是一个接口,它有几个具体的类实现。在反序列化过程中,如果我不知道该节点属于哪种类型的先验,我如何与Gson通信哪个具体类在反序列化节点时要实现?每个节点都有一个指定类型的成员字段。当对象是序列化形式时,有没有办法访问该字段,并以某种方式将类型传达给Gson?
谢谢!
答案 0 :(得分:44)
我建议为Node
s添加自定义JsonDeserializer:
Gson gson = new GsonBuilder()
.registerTypeAdapter(Node.class, new NodeDeserializer())
.create();
您将能够访问表示解串器方法中节点的JsonElement
,将其转换为JsonObject
,并检索指定类型的字段。然后,您可以基于此创建正确类型Node
的实例。
答案 1 :(得分:26)
您需要注册JSONSerializer和JSONDeserializer。您还可以通过以下方式为所有接口实现通用适配器:
这是我自己使用的实现,并且工作正常。
public class PropertyBasedInterfaceMarshal implements
JsonSerializer<Object>, JsonDeserializer<Object> {
private static final String CLASS_META_KEY = "CLASS_META_KEY";
@Override
public Object deserialize(JsonElement jsonElement, Type type,
JsonDeserializationContext jsonDeserializationContext)
throws JsonParseException {
JsonObject jsonObj = jsonElement.getAsJsonObject();
String className = jsonObj.get(CLASS_META_KEY).getAsString();
try {
Class<?> clz = Class.forName(className);
return jsonDeserializationContext.deserialize(jsonElement, clz);
} catch (ClassNotFoundException e) {
throw new JsonParseException(e);
}
}
@Override
public JsonElement serialize(Object object, Type type,
JsonSerializationContext jsonSerializationContext) {
JsonElement jsonEle = jsonSerializationContext.serialize(object, object.getClass());
jsonEle.getAsJsonObject().addProperty(CLASS_META_KEY,
object.getClass().getCanonicalName());
return jsonEle;
}
}
然后您可以为所有接口注册此适配器,如下所示
Gson gson = new GsonBuilder()
.registerTypeAdapter(IInterfaceOne.class,
new PropertyBasedInterfaceMarshal())
.registerTypeAdapter(IInterfaceTwo.class,
new PropertyBasedInterfaceMarshal()).create();
答案 2 :(得分:1)
据我所知,这不适用于非集合类型,或者更具体地说,使用具体类型进行序列化的情况,并且接口类型用于反序列化。也就是说,如果你有一个实现接口的简单类并且序列化具体类,那么指定要反序列化的接口,你将最终处于不可恢复的状态。
在上面的示例中,类型适配器是针对接口注册的,但是当您使用具体类进行序列化时,它将不会被使用,这意味着将永远不会设置CLASS_META_KEY数据。
如果您将适配器指定为分层适配器(从而告诉gson将其用于层次结构中的所有类型),您将最终处于无限循环中,因为序列化器将继续调用自身。
任何人都知道如何从接口的具体实现序列化,然后仅使用接口和InstanceCreator进行反序列化?
默认情况下,gson似乎会创建具体实例,但不会设置它的字段。
此处记录了问题:
http://code.google.com/p/google-gson/issues/detail?id=411&q=interface
答案 3 :(得分:0)
您必须使用Google Gson的TypeToken
课程。
你当然需要一个通用的类T来使其工作
Type fooType = new TypeToken<Foo<Bar>>() {}.getType();
gson.toJson(foo,fooType);
gson.fromJson(json, fooType);