我对抽象类的对象引用以及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 {
// ...
}
的Edge
来源和目标类型为Node
,被序列化为引用(@JsonIdentityReference(alwaysAsId = true)
):
public class Edge {
@JsonIdentityReference(alwaysAsId = true)
public Node source;
@JsonIdentityReference(alwaysAsId = true)
public Node target;
}
该图定义如下:
public class Graph {
public List<GreenNode> greenNodes = new ArrayList();
public List<RedNode> redNodes = new ArrayList();
public List<Edge> edges = new ArrayList();
}
JSON示例如下所示:
{
"greenNodes" : [ {
"id" : "g",
"content" : "green g",
"greenProperty" : "green"
} ],
"redNodes" : [ {
"id" : "r",
"content" : "red r",
"redProperty" : "red"
} ],
"edges" : [ {
"source" : "g",
"target" : "r"
} ]
}
使用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 {
然后,一切正常:
{
"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"
} ]
}
然后,一切正常。
真的有必要在引用的对象中包含类型信息以使引用起作用吗?如果没有类型信息,则可以加载具有红色和绿色节点(无边)的图形。进入边缘后,就不能。但是,边缘的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();
}
每个子类将自己指定为,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";
}
}
这样,JSON保持干净,但是Jackson可以解析id引用而没有任何问题。
本文收集自互联网,转载请注明来源。
如有侵权,请联系 [email protected] 删除。
我来说两句