1

Jackson 可以在 2.6.5 中为以下类反序列化 json,但在 2.8.8 中失败。

模型:

public static class Parent {
    public long id;
    public List<Child> children;
}

@RequiredArgsConstructor
public static class Child {
    public long childId;

    @NonNull
    @JsonIgnore
    public Parent parent;

    public Child() { }
}

JSON:

{
  "id": 1,
  "children": [
    {
      "childId": 2
    }
  ]
}

例外是:

com.fasterxml.jackson.databind.exc.UnrecognizedPropertyException: Unrecognized field "childId" (class Parent), not marked as ignorable (2 known properties: "children", "id"])

我发现由 lombok 创建的 Child 构造函数导致了这个错误。当我摆脱 lombok 注释或手动创建构造函数时,这种情况就不会发生了。无论哪种方式,它都应该使用无参数 Child() 构造函数。是什么导致了这个问题?

4

1 回答 1

3

Lombok 将注解添加@ConstructorProperties({"parent"})到生成的构造函数中。在 Jackson 2.8.8 中,这会导致构造函数被视为“委托创建者”。

委托创建者允许 Jackson 将一种类型对象的 json 反序列化为另一种类型的 Java 对象。

在这种情况下,由于 lombok 生成构造函数,@ConstructorProperties({"parent"}) Child(Parent parent) {...}Jackson 将尝试将子 json 反序列化为 Parent 对象,然后可以将其传递给构造函数以创建 Child。然后它抛出异常,因为 childId 不是 Parent 中的字段。

一种解决方法是ObjectMapper使用自定义配置用于反序列化 JSON,JacksonAnnotationIntrospector这样它就不会将构造函数解释为委托创建者。

ObjectMapper objectMapper = new ObjectMapper();
objectMapper.setAnnotationIntrospector(new JacksonAnnotationIntrospector().setConstructorPropertiesImpliesCreator(false));

更新

项目 lombok 的 1.16.20 版本确实默认 lombok.anyConstructor.suppressConstructorProperties 为 true,正如 Roel 在他的评论中指出的那样。这使得将 lombok 升级到最新版本是解决此问题的另一个方法。

于 2017-12-01T18:08:49.360 回答