在fastxml中,在反序列化json之后,如果枚举(带有JsonFormat.Shape.OBJECT)是类中的第一个属性,则其他字段为null。
为什么枚举应该是类中最后声明的属性,才能正确地反序列化其他字段?
也许这可能是fastxml中的错误?
示例类MyClass
:
public class MyClass {
// >>>
// >>> element field is null after deserialization
// >>>
private MyEnum option; // first
private String element; // --> null
// >>>
// >>> correctly deserialized if enum is last in order
// >>>
// private String element; // --> "elem"
// private MyEnum option; // last
public MyEnum getOption() {
return option;
}
public void setOption(MyEnum option) {
this.option = option;
}
public String getElement() {
return element;
}
public void setElement(String element) {
this.element = element;
}
}
示例枚举MyEnum
:
@JsonFormat(shape = JsonFormat.Shape.OBJECT)
public enum MyEnum {
FIRST;
@JsonProperty
public String getOption() {
return name();
}
@JsonCreator
public static MyEnum forValue(String option) {
return FIRST;
}
}
主测试类Main
的示例:
public class Main {
public static void main(String[] args) throws IOException {
ObjectMapper mapper = new com.fasterxml.jackson.databind.ObjectMapper();
mapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
MyClass myClass = new MyClass();
myClass.setElement("elem");
myClass.setOption(MyEnum.FIRST);
String serialized = mapper.writer().withDefaultPrettyPrinter().writeValueAsString(myClass);
System.out.println(String.format("serialized - %s", serialized));
MyClass deserialized = mapper.readValue(serialized, MyClass.class);
String deserializedResult = mapper.writerWithDefaultPrettyPrinter().writeValueAsString(deserialized);
System.out.println(String.format("deserialized - %s", deserializedResult));
}
}
反序列化后,输出显示字段为null
:
serialized - {
"option" : {
"option" : "FIRST"
},
"element" : "elem"
}
deserialized - {
"option" : {
"option" : "FIRST"
},
"element" : null
}
定单后的输出(MyClass
中未注释的行):
serialized - {
"element" : "elem",
"option" : {
"option" : "FIRST"
}
}
deserialized - {
"element" : "elem",
"option" : {
"option" : "FIRST"
}
}
答案 0 :(得分:1)
我无法告诉您这是否是一个错误,您可以调试并逐步执行代码以了解Jackson在这种情况下是如何“失败”的。您使用FAIL_ON_UNKNOWN_PROPERTIES
掩盖了问题,即使用String
作为forValue
工厂方法的参数类型。简而言之,Jackson在遍历JSON内容的令牌时会“陷入困境”。
要正确修复它,即不依赖订单,您有两种选择。首先,摆脱JsonFormat.Shape.OBJECT
形状以序列化枚举类型及其对应的@JsonCreator
。枚举的默认序列化/反序列化仍然使用其名称。
第二,如果您确实想要保留OBJECT形状,则需要更改@JsonCreator
方法以接收ObjectNode
,因为这是JSON所包含的,而不是{{1} }。从那里,您可以自己执行反序列化(假设您有更多的枚举常量)
String