我正在尝试使用杰克逊反序列化 json 对象并获得异常
`com.fasterxml.jackson.databind.exc.UnrecognizedPropertyException:
Unrecognized field "mobile" (class mypack.JacksonPhoneBuilder), not
marked as ignorable (2 known properties: , "phoneType", "value"]) at
[Source: %filelocation%; line: 8, column: 24] (through reference
chain:
mypack.JacksonAddressListBuilder["addresses"]->mypack.JacksonAddressBuilder["phones"]->mypack.JacksonPhoneBuilder["mobile"])`
这是对象:
{
"addresses": [
{
...
"phones": {
"mobile": "+01234567890"
}
},
...
]
}
电话.java:
@JsonDeserialize(builder = JacksonBuilder.class)
public class Phone {
protected String value;
protected Type type;
// setters and getters
}
我读过关于杰克逊枚举反序列化的信息,但有普通的枚举并且使用了地图。显然,领域"移动"不是在模型中表示,但它是枚举值,所以我如何反序列化它?
您的 JacksonPhoneBuilder 的工作方式与 Jackson 默认反序列化相同。问题是它能够以以下形式读取手机:
{
"type": "mobile",
"value": "+01234130000"
}
但是,在您的json对象中,电话表示为一个子对象,可以在Java中将其视为Map<PhoneType, String>
。一种可能的解决方案是使用从地图到列表的转换器(我假设一个地址中可能有很多手机)。
import com.fasterxml.jackson.databind.JavaType;
import com.fasterxml.jackson.databind.type.TypeFactory;
import com.fasterxml.jackson.databind.util.Converter;
public class PhoneConverter implements Converter<Map<PhoneType, String>, List<Phone>>{
public List<Phone> convert(Map<PhoneType, String> phonesMap) {
List<Phone> phones = new ArrayList<Phone>();
for (PhoneType phoneType : phonesMap.keySet()) {
phones.add(new Phone(phoneType, phonesMap.get(phoneType)));
}
return phones;
}
public JavaType getInputType(TypeFactory typeFactory) {
return typeFactory.constructMapLikeType(Map.class, PhoneType.class, String.class);
}
public JavaType getOutputType(TypeFactory typeFactory) {
return typeFactory.constructCollectionLikeType(List.class, Phone.class);
}
}
然后在地址类中:
public class Address {
@JsonDeserialize(converter = PhoneConverter.class)
protected List<Phone> phones;
}
请注意,它不会与您的构建器一起使用,但如果您不执行任何其他自定义反序列化,那么您不需要它们 - 您可以依赖 Jackson 的默认行为。