【问题标题】:Jackson custom deserializer for one field with polymorphic typesJackson 自定义反序列化器,用于具有多态类型的一个字段
【发布时间】:2015-02-05 17:41:08
【问题描述】:

更新:

我尝试在jackson源代码中调试,在方法中发现

deserialize(JsonParser jp, DeserializationContext ctxt)

SettableBeanProperty.java

_valueTypeDeserializer 不为空时,它永远不会使用_valueDeserializer

TypeDeserializer 应该比 ValueDeserializer 更合适的假设是否正确?


我正在尝试使用@JsonDeserialize 为具有多态类型的一个字段定义自定义反序列化器。我可以成功分别使用@JsonDeserialize@JsonTypeInfo。但是当我一起使用它们时,似乎@JsonDeserialize 注释被忽略了。

这是我的类层次结构:

1. 定义.java

@JsonTypeInfo(use = JsonTypeInfo.Id.NAME, include = JsonTypeInfo.As.PROPERTY, property = "type", defaultImpl = Definition.class)
@JsonSubTypes({@Type(value = Definition.class, name = "normal"),
    @Type(value = FormDefinition.class, name = "form")})
public class Definition {

    private String name;
    private String description;

    // getter&setter for name&description
}

2. FormDefinition.java

public class FormDefinition extends Definition {

    private String formName;
    @JsonDeserialize(using = DefinitionDeserializer.class)
    private Definition definition;

    public String getFormName() {
        return formName;
    }

    public void setFormName(String formName) {
        this.formName = formName;
    }

    public void setDefinition(Definition definition) {
        this.definition = definition;
    }
}

3。定义Deserializer.java

public class DefinitionDeserializer extends JsonDeserializer<Definition> {

    @Override 
    public Definition deserialize(JsonParser jsonParser, DeserializationContext deserializationContext)
            throws IOException, JsonProcessingException {
        Definition definition = new Definition();
        String name = jsonParser.readValueAs(String.class);
        definition.setName(name);
        return definition;
    }
}

示例主要

public class App 
{
    public static void main( String[] args ) throws IOException {
        String sampleData = "{\"type\":\"form\",\"definition\":\"super\",\"formName\":\"FormName\"}";
        ObjectMapper mapper = new ObjectMapper();
        Definition definition = mapper.readValue(sampleData, Definition.class);
        System.out.println(definition);
    }
}

运行示例主应用程序将引发异常:

Exception in thread "main" com.fasterxml.jackson.databind.JsonMappingException: Can not instantiate value of type [simple type, class jp.co.worksap.model.Definition] from String value ('super'); no single-String constructor/factory method (through reference chain: jp.co.worksap.model.FormDefinition["definition"])

这似乎使用AsPropertyTypeDeserializer 来反序列化FormDefinition 类的definition 字段,而不是带注释的反序列化器DefinitionDeserializer

我认为这里的棘手部分是我想使用自定义反序列化器的字段也是Definition 类型,它使用@JsonTypeInfo 来解决多态问题。

有什么方法可以一起使用吗?谢谢。

【问题讨论】:

    标签: java json polymorphism jackson deserialization


    【解决方案1】:

    Jackson 在选择使用哪个 Deserializer 之前处理 @JsonTypeInfo,可能是因为 Deserializer 的选择通常取决于类型。但是,您可以像指定自定义反序列化器一样轻松地在每个字段的基础上禁用此功能 - 通过直接注释字段:

    @JsonDeserialize(using = DefinitionDeserializer.class)
    @JsonTypeInfo(use = JsonTypeInfo.Id.NONE)
    private Definition definition;
    

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 2016-02-12
      • 1970-01-01
      • 1970-01-01
      • 2014-05-01
      • 1970-01-01
      • 2018-03-06
      • 1970-01-01
      • 2021-12-20
      相关资源
      最近更新 更多