我对抽象类和 JSON 序列化和反序列化的对象引用有问题。抽象的问题如下所示:
我有一个由节点和边组成的图。每条边连接两个节点。节点可以是红色和绿色的。因此,有一个抽象类Node和两个派生类RedNode和GreenNode。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 获得。
当前的解决方案是包含虚假类型信息。完整代码位于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 引用。
| 归档时间: |
|
| 查看次数: |
2495 次 |
| 最近记录: |