3

我在使用 GSON 序列化和反序列化我的对象结构时遇到问题。为了描述这个问题,我必须稍微描述一下我的类结构:

我有一个java抽象类,我们把它命名为“A”。还有抽象类“BA”、“CA”、“DA”,它们扩展了“A”类。他们每个人都有自己的构造函数,其中非非参数。最后,有几个(很多!)类扩展了“BC”、“CA”或“DA”。这些“底层”类的实例保存在“ArrayList”列表中。

现在,我正在尝试“jsonize”该数组列表。为了创建 Json 字符串,我使用以下代码:

Gson gs = new Gson();
Type listOfTestObject = new TypeToken<List<A>>(){}.getType();
String rez = gs.toJson(getListOfAs(), listOfTestObject);

我正在尝试使用这个(在另一个类中)反序列化该json:

Type listOfTestObject = new TypeToken<ArrayList<A>>(){}.getType();
ArrayList<A> listOfAs = gs.fromJson(jsonREZString, listOfTestObject);

但上面的代码抛出了这个:

Unable to invoke no-args constructor for class packagename.A. Register an InstanceCreator with Gson for this type may fix this problem.

现在,我在“A”类中创建了一个非参数构造函数,但没有运气。我已经阅读了“InstanceCreator”,但看起来我必须为每个扩展“A”的具体类创建一个“InstanceCreator”!对?我做不到,因为我有很多(很多!)通过“BA”、“CA”或“DA”扩展“A”的类。

我错过了什么?如何在不为每种类型添加自定义反序列化代码的情况下简单地反序列化(序列化似乎很好)这个复杂的结构?

4

1 回答 1

3

事实上,您可能会遇到 2 个不同的问题。

1)您具有多态类型,因此您可能希望将对象序列化为其具体类型而不是 A。2)您希望反序列化为不提供 arg ctrs 的具体类型。

Gson 不支持 1 和 2,有 1 的扩展,但我从未使用过。

也许Genson解决了你的问题,它同时支持 1 和 2。

Genson genson = new Genson.Builder()
                            // enables polymorphic types support
                            .setWithClassMetadata(true)
                            // enables no arg support
                            .setWithDebugInfoPropertyNameResolver(true)
                          .create();

// will look like: [{"@class": "com.xxx.SomeConcreteClass", ...}, {"@class": "com.XXX.OtherClass"}]
String json = genson.serialize(getListOfAs());
List<A> listOfA = genson.deserialize(json, new GenericType<List<A>>() {});

您不需要在序列化过程中指定类型,除非您希望输出中仅存在父字段。

例如:genson.serialize(getListOfAs(), GenericType>() {}) 将仅序列化来自 A 的属性,您还可以通过在构建器上设置 setUseRuntimeTypeForSerialization(true) 来强制 genson 始终使用运行时类型。

此外,如果您不希望 impl 详细信息在 json 表示中泄漏,您可以为您的类型定义别名 (builder.addAlias("someAlias", SomeClass.class),它们将被使用而不是完整的包 + 类名。

于 2013-06-05T08:52:35.160 回答