杰克逊枚举反序列化



我正在尝试使用杰克逊反序列化 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 的默认行为。

最新更新