问题描述
我正在尝试在 Java 中使用 Jackson 反序列化遇到一些麻烦。我已经提出了 2 个解决方案,但我无法解决问题。问题?我得到了重复属性的结果,这是杰克逊反序列化后重复的字段。 (我的问题和这个问题完全一样:Avoid duplicate field generated by JsonTypeInfo in Jackson,当时没人能给你答案)
首先,我有以下课程:
@JsonIgnoreProperties(ignoreUnkNown = true)
public class Instance {
@JsonProperty("id")
private String id;
@JsonProperty("name")
private String name;
@JsonProperty("type")
private InstanceType type;
}
我正在尝试做的只是实例化一个“实例”类型的对象,保存并读取它。使用解决方案 2,对象以重复的类型保存(类型显示为包含 'name'、'firs_type' 或 'second_type' 的数组)取决于我创建的内容。使用解决方案 1,我可以保存对象,但是当我尝试读取它时,我遇到了 jackson 异常转换。
解决方案 1:
@JsonDeserialize(using = InstanceTypeDeserializer.class)
public interface InstanceType {
String value();
}
@JsonDeserialize(as = HardInstanceType.class)
public enum HardInstanceType implements InstanceType {
FirsT_TYPE("first_type"),SECOND_TYPE("second_type")
private String value;
HardInstanceType(String value) {
this.value = value;
}
@JsonValue
public String value() {
return value;
}
}
@JsonDeserialize(as = SoftInstanceType.class)
public enum SoftInstanceType implements InstanceType {
//.. types implementaion similar as HardInstanceType
}
public class InstanceTypeDeserializer extends JsonDeserializer<InstanceType> {
@Override
public InstanceType deserialize(JsonParser jp,DeserializationContext ctxt) throws IOException,JsonProcessingException {
ObjectMapper mapper = (ObjectMapper) jp.getCodec();
ObjectNode root = (ObjectNode) mapper.readTree(jp);
if(root.get("name").asText().equals("hard")) {
return mapper.readValue(root.toString(),HardInstanceType.class);
} else {
return mapper.readValue(root.toString(),SoftInstanceType.class);
}
}
}
此解决方案的问题在于,当我尝试存储数据并映射到类时,出现以下错误:
异常解析json: com.fasterxml.jackson.databind.JsonMappingException:类 com.fasterxml.jackson.databind.node.TextNode 无法转换为类 com.fasterxml.jackson.databind.node.ObjectNode (com.fasterxml.jackson.databind.node.TextNode 和 com.fasterxml.jackson.databind.node.ObjectNode 在未命名的模块中 加载器 org.springframework.boot.loader.Launchedurlclassloader @1a3e8e24)(通过参考链: java.util.ArrayList[0]->com.project.package.xxxx.Instance["type"])
解决方案 2
@JsonTypeInfo(
use = JsonTypeInfo.Id.NAME,include = JsonTypeInfo.As.PROPERTY,property = "name")
@JsonSubTypes({
@JsonSubTypes.Type(value = HardInstanceType.class,name = "hard") })
public interface InstanceType {
String value();
}
这个解决方案的问题是,当我保存数据时,当我创建一个实例对象并将其存储在数据存储中时,我得到以下内容:
"id": "1","name": "hard","type": [
"hard","first_type"
]
什么是不正确的,类型应该只存储“first_type”(解决方案 1 存储了什么,但我看不懂哈哈)。
当然,Instace 类更复杂,字段也更多,我在这里简化一下,仅作为示例。
我需要这方面的帮助,非常感谢您。
解决方法
终于可以解决问题了。 我发布这个以防万一其他人需要它。
-
向我的 HardInstanceType 类添加一个属性。
公共枚举 HardInstanceType 实现 InstanceType {
FIRST_TYPE("first_type"),SECOND_TYPE("second_type"); private String value; public String hardTypeIdentifierSer = "hardTypeIdentifierSer"; HardInstanceType(String value) { this.value = value; } @JsonValue public String value() { return value; }
}
-
然后,在解串器中:
公共类 InstanceTypeDeserializer 扩展 JsonDeserializer {
@Override public InstanceType deserialize(JsonParser jp,DeserializationContext ctxt) throws IOException,JsonProcessingException { TreeNode node = jp.readValueAsTree(); if (node.get("hardTypeIdentifierSer") != null) { return jp.getCodec().treeToValue(node,HardInstanceType.class); }
}