在fasterxml中,反序列化json后,如果enum(带有JsonFormat.Shape.OBJECT(是类中的第一个属性,则其他字段为null。
为什么枚举应该是类中最后一个声明的属性才能正确地反序列化其他字段?
也许这可能是fasterxml中的一个错误?
示例类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"
}
}
我不能告诉你这是否是一个bug,你可以调试并逐步完成代码,以了解Jackson在这种情况下是如何"失败"的。使用FAIL_ON_UNKNOWN_PROPERTIES
隐藏了问题,即使用String
作为forValue
工厂方法的参数类型。简而言之,Jackson在遍历JSON内容的令牌时遇到了"麻烦"。
要正确修复它,即不依赖订单,你有几个选择。首先,去掉用于序列化枚举类型及其对应的@JsonCreator
的JsonFormat.Shape.OBJECT
形状。枚举的默认序列化/反序列化无论如何都要使用其名称。
其次,如果您真的想保持OBJECT形状,您需要更改@JsonCreator
方法以接收ObjectNode
,因为这是JSON包含的内容,而不是String
。从那里,您可以自己执行反序列化(假设您有更多的枚举常量(
@JsonCreator
public static MyEnum forValue(ObjectNode object) {
return MyEnum.valueOf(object.get("option").asText());
}