如何反序列化引用 Jackson 中的抽象类型的 JSON

kop*_*por 5 jackson

我对抽象类和 JSON 序列化和反序列化的对象引用有问题。抽象的问题如下所示:

我有一个由节点和边组成的图。每条边连接两个节点。节点可以是红色和绿色的。因此,有一个抽象类Node和两个派生类RedNodeGreenNode。ANode取一个id( @JsonIdentityInfo(generator = ObjectIdGenerators.PropertyGenerator.class, property = "id")):

@JsonSubTypes({
    @JsonSubTypes.Type(value = GreenNode.class, name = "GreenNode"),
    @JsonSubTypes.Type(value = RedNode.class, name = "RedNode")
})
@JsonIdentityInfo(generator = ObjectIdGenerators.PropertyGenerator.class, property = "id")
public abstract class Node {
    public String id;
}

public class RedNode extends Node {
    // ...
}

public class GreenNode extends Node {
    // ...
}
Run Code Online (Sandbox Code Playgroud)

AnEdge有一个 source 和一个 target 类型Node,它们被序列化为引用 ( @JsonIdentityReference(alwaysAsId = true)):

public class Edge {
    @JsonIdentityReference(alwaysAsId = true)
    public Node source;
    @JsonIdentityReference(alwaysAsId = true)
    public Node target;
}
Run Code Online (Sandbox Code Playgroud)

该图定义如下:

public class Graph {
    public List<GreenNode> greenNodes = new ArrayList();
    public List<RedNode> redNodes = new ArrayList();
    public List<Edge> edges = new ArrayList();
}
Run Code Online (Sandbox Code Playgroud)

JSON 示例如下所示:

{
  "greenNodes" : [ {
    "id" : "g",
    "content" : "green g",
    "greenProperty" : "green"
  } ],
  "redNodes" : [ {
    "id" : "r",
    "content" : "red r",
    "redProperty" : "red"    
  } ],
  "edges" : [ {
    "source" : "g",
    "target" : "r"
  } ]
}
Run Code Online (Sandbox Code Playgroud)

使用ObjectMapper无法读取:

无法构造 com.github.koppor.jsonidentityissue.model.Node 的实例:抽象类型要么需要映射到具体类型,要么具有自定义反序列化器,要么包含其他类型信息

错误位置是“行:13,列:16”。因此,它在边缘的 id 处被击中。节点本身已正确序列化。

一种解决方法是在 json 中添加类型信息:

@JsonTypeInfo(
      use = JsonTypeInfo.Id.NAME,
      include = JsonTypeInfo.As.PROPERTY,
      property = "type")
public abstract class Node {
Run Code Online (Sandbox Code Playgroud)

然后,一切正常:

{
  "greenNodes" : [ {
    "id" : "g",
    "type" : "GreenNode",
    "content" : "green g",
    "greenProperty" : "green"
  } ],
  "redNodes" : [ {
    "id" : "r",
    "type" : "RedNode",
    "content" : "red r",
    "redProperty" : "red"    
  } ],
  "edges" : [ {
    "source" : "g",
    "target" : "r"
  } ]
}
Run Code Online (Sandbox Code Playgroud)

然后,一切正常。

是否真的有必要在引用的对象中包含类型信息才能使引用正常工作?如果没有类型信息,可以加载带有红色和绿色节点(并且没有边)的图。边缘进来后,就不能了。但是,边的 JSON 仅包含一个 id。引用的对象已经被解析。

我真的很喜欢摆脱@JsonTypeInfo注释。有没有办法拥有一个干净的 JSON?

完整示例可在https://github.com/koppor/jackson-jsonidentityreference-issue/tree/issue 获得

kop*_*por 3

当前的解决方案是包含虚假类型信息。完整代码位于https://github.com/koppor/jackson-jsonidentityreference-issue

获取Node一个现有的属性type,未写入:

@JsonTypeInfo(
    use = JsonTypeInfo.Id.NAME,
    include = JsonTypeInfo.As.EXISTING_PROPERTY,
    property = "type")
public abstract class Node {
    @JsonIgnore
    public abstract String getType();
}
Run Code Online (Sandbox Code Playgroud)

每个子类将自身指定为defaultImpl并提供以下实现getType

@JsonTypeInfo(
    use = JsonTypeInfo.Id.NAME,
    include = JsonTypeInfo.As.EXISTING_PROPERTY,
    property = "type",
    defaultImpl=GreenNode.class)
public class GreenNode extends Node {
    @Override
    public String getType() {
        return "GreeNode";
    }
}
Run Code Online (Sandbox Code Playgroud)

这样,JSON 就保持干净,但 Jackson 可以毫无问题地解析 id 引用。