java - 使用 Jackson 对 JSON 文件进行多态反序列化

标签 java json jackson polymorphism deserialization

根据 JSON 文件的内容,我想将其反序列化为父类(super class)或子类。

如果它看起来像这样,则应该反序列化为父类(super class):

{
   "id":"123",
   "title":"my title",
   "body":"my body"
}

或者子类,如果它看起来像这样:

{
   "id":"123",
   "title":"my title",
   "body":"my body",
   "tags":["tag1", "tag2"]
}

所以唯一的区别是 tags 数组,它应该被反序列化为字符串数组。 但如果我通过 POST 请求触发 Jersey (Dropwizard) 中的反序列化,它会返回 {"code":400,"message":"Unable to process JSON"}

这是父类(super class):

@JsonTypeInfo(use = JsonTypeInfo.Id.NAME)
@JsonSubTypes({ @JsonSubTypes.Type(name = "subdocument", value = SubDocument.class) })
public class SuperDocument {

private String id;
private String title;
private String body;

public SuperDocument() {

}

@JsonCreator
public SuperDocument(@JsonProperty("id") String id, @JsonProperty("title") String title, @JsonProperty("body") String body) {
    this.id = id;
    this.title = title;
    this.body = body;
}

@JsonProperty("id")
public String getId() {
    return id;
}

@JsonProperty("id")
public void setId(String id) {
    this.id = id;
}

... the other getters and setters ...
}

这是子类:

@JsonTypeName("subdocument")
public class SubDocument extends SuperDocument {

private String[] tags;

public SubDocument() {

}

@JsonCreator
public SubDocument(@JsonProperty("id") String id, @JsonProperty("title") String title, @JsonProperty("body") String body, @JsonProperty("tags") String[] tags) {
    super(id, title, body);
    this.tags = tags;
}

@JsonProperty("tags")
public String[] getTags() {
    return tags;
}

@JsonProperty("tags")
public void setTags(String[] tags) {
    this.tags = tags;
}
}

你知道我做错了什么吗?

最佳答案

JsonTypeInfo 需要一个可以标识您的子类/父类(super class)的属性。例如:

{
   "id":"123",
   "title":"my title",
   "body":"my body",
   "type":"superdocument"
}

{
   "id":"123",
   "title":"my title",
   "body":"my body",
   "tags":["tag1", "tag2"],
   "type":"subdocument"
}

然后修改SuperDocument注释,如下所示。

@JsonTypeInfo(use = JsonTypeInfo.Id.NAME,property="type")
@JsonSubTypes({ @JsonSubTypes.Type(name = "subdocument", value = SubDocument.class),@JsonSubTypes.Type(name = "superdocument", value = SuperDocument.class) })

public class SuperDocument {

}

如果您不想引入额外的属性“type”,那么您可能必须编写自定义类型解析器和类型反序列化器,如下所示。

    public class DocumentTypeResolver extends StdTypeResolverBuilder {
    @Override
    public TypeDeserializer buildTypeDeserializer(
            final DeserializationConfig config, final JavaType baseType, final Collection<NamedType> subtypes) {
        return new DocumentDeserializer(baseType, null,
                _typeProperty, _typeIdVisible, _defaultImpl);
    }
}

自定义类型反序列化器

    public static class DocumentDeserializer extends AsPropertyTypeDeserializer {

    public DocumentDeserializer(final JavaType bt, final TypeIdResolver idRes, final String typePropertyName, final boolean typeIdVisible, final Class<?> defaultImpl) {
        super(bt, idRes, typePropertyName, typeIdVisible, defaultImpl);
    }

    public DocumentDeserializer(final AsPropertyTypeDeserializer src, final BeanProperty property) {
        super(src, property);
    }

    @Override
    public TypeDeserializer forProperty(final BeanProperty prop) {
        return (prop == _property) ? this : new DocumentDeserializer(this, prop);
    }

    @Override
    public Object deserializeTypedFromObject(final JsonParser jp, final DeserializationContext ctxt) throws IOException {
        JsonNode node = jp.readValueAsTree();
        Class<?> subType =null;
        JsonNode tags = node.get("tags");
        if (tags == null) {
             subType=SuperDocument.class;
        } else {
            subType=SubDocument.class;
        }
        JavaType type = SimpleType.construct(subType);
        JsonParser jsonParser = new TreeTraversingParser(node, jp.getCodec());
        if (jsonParser.getCurrentToken() == null) {
            jsonParser.nextToken();
        }

        JsonDeserializer<Object> deser = ctxt.findContextualValueDeserializer(type, _property);
        return deser.deserialize(jsonParser, ctxt);

    }

}

现在注释您的 SuperDocument 类,如下所示

@JsonTypeInfo(use = JsonTypeInfo.Id.NONE)
@JsonTypeResolver(DocumentTypeResolver.class)
public class SuperDocument {

}

关于java - 使用 Jackson 对 JSON 文件进行多态反序列化,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43020466/

相关文章:

java - 可以说学习 Java 会让我成为更好的 PHP OOP 开发人员吗?

java - 将参数传递给脚本 - 参数是位于上一个目录的配置文件

java - 使用流过滤java中的内部和外部列表

mysql - 检查包含 json 字符串的列是否具有特定值

java - Jackson 会抛出循环异常吗?

java - 无法使用 Iterable 从 java 中的自定义链表检索数据

javascript - 在 React JavaScript 中显示/映射嵌套 JSON 字段

android - 如何使用以下 JSON 响应在另一个 recyclerView 中显示一个 recyclerView

java - Jackson 将 Object 序列化为 JSON 到 base64(没有无限循环)

java - JsonManagedReference 与 JsonBackReference