当前位置: 首页 > 知识库问答 >
问题:

为多态类型层次结构编写自定义反序列化器Jackson

郝承悦
2023-03-14

我正在用Jackson反序列化在Java中进行继承实验。我有一个基类:

@Getter //Lombok @Getter
@JsonTypeInfo(use = JsonTypeInfo.Id.NAME, property = "type", include = JsonTypeInfo.As.PROPERTY)
@JsonSubTypes(value = {
        @JsonSubTypes.Type(value=ClassA.class, name = "classA")
})
public abstract class BaseClass {
   private List<String> fields;

   @JsonCreator
   public BaseClass(@JsonProperty("fields") final List<String> fields) {
     this.fields = fields;
   }
}

ClassA也是抽象的

@Getter
@JsonTypeInfo(use = JsonTypeInfo.Id.NAME, property = "typeA", include = JsonTypeInfo.As.PROPERTY)
@JsonSubTypes(value = {
        @JsonSubTypes.Type(value=SubClassA.class, name = "subclassA")
})
public abstract class ClassA extends BaseClass{
   private String mode;

   @JsonCreator
   public ClassA(@JsonProperty("fields") final List<String> fields, @JsonProperty("mode") String mode) {
       super(fields);
       this.mode = mode;
   }
}

我的子类:

public class SubClassA extends ClassA {
   private String dummyField;

   public SubClassA(@JsonProperty("fields") final List<String> fields, @JsonProperty("mode") String mode,
     @JsonProperty("dummyField") String dummyField) {
       super(fields, mode);
       this.dummyField = dummyField;
   }

}

如果我以以下形式传入的JSON:

{
  "type": "classA",
  "typeA": "subclassA",
  "mode": "testingMode",
  "fields": ["1", "2"],
  "dummyField": "dummy"
}

我得到一个错误,无法构造ClassA的实例(不存在像默认构造一样的创建者):抽象类型要么需要映射到具体类型,要么需要自定义反序列化器,要么包含我遇到的其他类型信息https://github.com/FasterXML/jackson-databind/issues/374这说明这是杰克逊的一个已知问题。我该如何为此编写自定义反序列化程序。

在A班,我尝试这样做:

@JsonDeserialize(using = ClassADeserializer.class)

ClassADeserializer是:

public class ClassADeserializer extends StdDeserializer<ClassA> {
    private final JsonDeserializer<?> defaultDeserializer;
    public ClassADeserializer(JsonDeserializer<?> defaultDeserializer) {
        super(ClassA.class);
        this.defaultDeserializer = defaultDeserializer;
    }

    @Override public ClassA deserialize(final JsonParser jsonParser, final DeserializationContext deserializationContext) throws IOException, JsonProcessingException {
       return (ClassA) defaultDeserializer.deserialize(jsonParser, deserializationContext);
    }

这显然不起作用。我该如何为此编写自定义反序列化程序?

共有1个答案

步浩壤
2023-03-14

您传入json"type":"class A",...这意味着jackson首先尝试创建ClassA的实例...在反序列化过程中,jackson搜索@JsonCreator构造函数首先...如果@JsonCreator丢失或无法调用@JsonCreator构造函数,则jackson使用默认构造函数和调用setter方法创建对象...在您的ClassA@JsonCreator构造函数中具有2个参数,但jackson使用3个参数调用...所以它失败了。然后杰克逊调用默认构造函数来创建实例。但默认构造函数也丢失了...这就是为什么你会得到这个错误:无法构造ClassA的实例(没有Creator,如默认构造,存在)...

当你想反序列化到子类时,你需要在子类中使用JsonCreator,然后你需要使用JsonIgnoreProperties来忽略属性,这样jackson就可以创建子类的实例而不是子类。。。。

尝试使用下面的SubClassA

@Getter
@JsonIgnoreProperties(ignoreUnknown = true)
public class SubClassA extends ClassA {
    private String dummyField;

    @JsonCreator
    public SubClassA(@JsonProperty("fields") final List<String> fields, @JsonProperty("mode") String mode,
                     @JsonProperty("dummyField") String dummyField) {
        super(fields, mode);
        this.dummyField = dummyField;
    }

}
 类似资料:
  • 问题内容: 我在 .NET for WinRT(C#)中 ,我想将JSON字符串反序列化为,然后将字典值稍后转换为实际类型。JSON字符串可以包含对象层次结构,我也希望在其中包含子对象。 这是应该能够处理的示例JSON: 我尝试使用 DataContractJsonSerializer 这样做: 实际上,这对于第一个级别是可行的,但是 “父母” 只是一个不能强制转换为的对象: 然后,我尝试使用 J

  • I'va是一个OID接口,可以由许多具体类型实现: 现在我有一个具有两个字段的对象,一个使用抽象接口类型(OID)定义,另一个使用具体类型(MyOID)定义 我想使用jackson以不同的方式序列化/反序列化字段,无论它们是使用抽象接口类型还是具体类型定义的: 注意,被序列化,包括类型信息(多态序列化),而被序列化为文本 为此,我将OID接口注释为: 并为每个具体类型分配了类型id: 最后,对容器

  • 有了这个数据模型... TestClass.kt 我的目标是反序列化以下json值: 我在这里看到了解决方案并尝试了这个,但是我的“反序列化”方法中的断点似乎永远不会命中。 选项Deserializer.java TestDeserialization.kt 建筑gradle(用于版本信息)

  • 问题内容: 如果我有这样的类结构: 还有另一种反序列化的方法吗?在父类上使用此注释: 我不想强迫我的API的客户包括反序列化一个子类。 杰克逊不是使用,而是提供了一种方法来注释子类并通过唯一属性将其与其他子类区分开来?在上面的示例中,这类似于“如果JSON对象将其反序列化为,如果它将其反序列化为”。 问题答案: 感觉像是应该使用的东西,但是我在文档中进行了选择,可以提供的任何属性似乎都不符合您所描

  • 我看过jackson反序列化@JsonTypeInfo的一个例子,那就是: 我试过了,效果很好。现在的问题是,在示例类中,Cat和Dog是从Animal中引用的,我想避免这种情况。有没有一种方法可以将类型绑定从类动物中移除,并且仍然进行反序列化工作?谢谢

  • 想象一下以下场景: 我想为Foo编写一个自定义的Jackson反序列化程序。为了做到这一点(例如,为了反序列化具有Foo的类 如何编写这样的反序列化程序?这应该是可能的,因为Jackson是通过键入集合和地图来实现的。 澄清: 似乎有两个部分可以解决这个问题: 1)获取内声明的属性类型并使用它来反序列化