无法反序列化具有最终字段的最简单对象 bu 杰克逊



我只想序列化和反序列化简单的不可变对象,但不明白为什么我不能使用 Jackson 来做到这一点

import java.io.IOException;
import com.fasterxml.jackson.databind.ObjectMapper;
import lombok.Value;
public class TestApplication {
    @Value
    static class Test {
        private final String a;
    }
    public static void main(String[] args) throws IOException {
        ObjectMapper objectMapper = new ObjectMapper();
        String res = objectMapper.writeValueAsString(new Test("test"));
        System.out.println(res);
        System.out.println(objectMapper.readValue(res, Test.class));
    }
}

它失败并出现异常:

Exception in thread "main" com.fasterxml.jackson.databind.exc.MismatchedInputException: Cannot construct instance of `com.devchallange.rogatakopita.RogatakopitaApplication$Test` (although at least one Creator exists): cannot deserialize from Object value (no delegate- or property-based Creator)
 at [Source: (String)"{"a":"test"}"; line: 1, column: 2]
    at com.fasterxml.jackson.databind.exc.MismatchedInputException.from(MismatchedInputException.java:63)
    at com.fasterxml.jackson.databind.DeserializationContext.reportInputMismatch(DeserializationContext.java:1343)
    at com.fasterxml.jackson.databind.DeserializationContext.handleMissingInstantiator(DeserializationContext.java:1032)
    at com.fasterxml.jackson.databind.deser.BeanDeserializerBase.deserializeFromObjectUsingNonDefault(BeanDeserializerBase.java:1297)
    at com.fasterxml.jackson.databind.deser.BeanDeserializer.deserializeFromObject(BeanDeserializer.java:326)
    at com.fasterxml.jackson.databind.deser.BeanDeserializer.deserialize(BeanDeserializer.java:159)
    at com.fasterxml.jackson.databind.ObjectMapper._readMapAndClose(ObjectMapper.java:4013)
    at com.fasterxml.jackson.databind.ObjectMapper.readValue(ObjectMapper.java:3004)
    at com.devchallange.rogatakopita.RogatakopitaApplication.main(RogatakopitaApplication.java:29)

我知道答案应该很容易,但这就像最常见的情况应该从盒子里工作,不是吗?

在设计不可变类时,Jackson 应该使用所有必需的参数调用构造函数。

如果您没有使用龙目岛,

  • @JsonCreator 注释您希望杰克逊调用的构造函数。
  • @JsonProperty 注释构造函数参数。如果要跳过此操作,可以添加使用ParameterNamesModule扩展。

例:

static class Test {
    private final String a;
    public Test() {
        a = "default";
    }
    @JsonCreator // Jackson will use this constructor during deserialization
    public Test(@JsonProperty("a") String a) { // @JsonProperty can be skipped if you use ParameterNamesModule annotation
        this.a = a;
    }
    // Getter for A
}

龙目岛的@Value注解只产生一个全参数构造函数,我们需要用@JsonCreator来注解。这可以通过使用 @AllArgsConstructor(onConstructor = @__(@JsonCreator)) 注释类来完成。

由于构造函数是自动生成的,我们将无法使用 @JsonProperty 注释参数,因此,您必须使用 ParameterNamesModule .

包含以下更改的代码片段:

public class TestApplication {
    @Value
    @AllArgsConstructor(onConstructor = @__(@JsonCreator))
    static final class Test {
        private final String a;
    }
    public static void main(String[] args) throws IOException {
        ObjectMapper objectMapper = new ObjectMapper();
        objectMapper.registerModule(new ParameterNamesModule());
        String res = objectMapper.writeValueAsString(new Test("test"));
        System.out.println(res);
        System.out.println(objectMapper.readValue(res, Test.class));
    }
}

以下是参数名称模块的 maven 依赖关系。

 <dependency>
      <groupId>com.fasterxml.jackson.module</groupId>
      <artifactId>jackson-module-parameter-names</artifactId>
      <version>2.9.8</version>
      <scope>compile</scope>
 </dependency>

在java中,必须在构造函数中设置最终属性,并且只能在以后读取。但杰克逊后来试图改变这一属性。所以你不能用杰克逊来填充最后一个字段。

最新更新