11

假设有一个抽象类,sayA和两个非抽象子类,sayA1A2。我想通过使用GSON library从 json 格式“反序列化”它们。

例如,我得到一个对象数组A

int n = ...;
A[] list = new A[n];
A[0] = new A1(....);
A[1] = new A2(....);
...

有人将其转换为 JSON 字符串,如下所示:

String json = (new Gson()).toJson(list);

最后,如果我尝试反序列化如下

A[] deserializedList =  (new Gson()).fromJson(json, A[].class);

然后我得到了一个错误,因为 GSON 默认反序列化器找到了一个抽象类(即A)并且它无法猜测子类类型。

我该如何解决这个问题?

PS:我阅读了自定义反序列化器,但我不明白在这种情况下如何使用它们。

4

1 回答 1

19

Axxiss链接之后,下面是答案。必须提供自定义序列化器/反序列化器。

public class AClassAdapter  implements JsonSerializer<A>, JsonDeserializer<A> {
  @Override
  public JsonElement serialize(A src, Type typeOfSrc, JsonSerializationContext context) {
      JsonObject result = new JsonObject();
      result.add("type", new JsonPrimitive(src.getClass().getSimpleName()));
      result.add("properties", context.serialize(src, src.getClass())); 
      return result;
  }


  @Override
  public A deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context)
        throws JsonParseException {
    JsonObject jsonObject = json.getAsJsonObject();
    String type = jsonObject.get("type").getAsString();
    JsonElement element = jsonObject.get("properties");

    try {            
        String fullName = typeOfT.getTypeName();
        String packageText = fullName.substring(0, fullName.lastIndexOf(".") + 1);

        return context.deserialize(element, Class.forName(packageText + type));
    } catch (ClassNotFoundException cnfe) {
        throw new JsonParseException("Unknown element type: " + type, cnfe);
    }
  }
}

然后序列化如下:

GsonBuilder gson = new GsonBuilder();
gson.registerTypeAdapter(A.class, new ATypeAdapter());
String json = gson.create().toJson(list);

并给定 json 字符串,反序列化为:

GsonBuilder gson = new GsonBuilder();
gson.registerTypeAdapter(A.class, new ATypeAdapter());

return gson.create().fromJson(json, A[].class);
于 2013-06-01T13:07:29.223 回答