自定义 Jackson 反序列化器不会注入 @Autowired 字段

时间:2021-03-20 16:00:32

标签: java json spring spring-boot jackson

我正在尝试将我的配置注入我的自定义 StdDeserializer。然而问题是,即使我将该类标记为 @Component,该字段的值也从未注入。注入在应用程序的其他地方没有任何问题。

因此,我得出的结论是,问题出在反序列化器的工作方式上,因为它没有被我们实例化,而是像下面的例子一样:

ClassToDeserialize myClass = new ObjectMapper().readValue(mockJson, ClassToDeserialize.class);

如您所见,我的自定义反序列化器 ClassToDeserializeDeserializer 没有明确使用,因此它使用带有 @JsonDeserialize(using = ClassToDeserialize.class) 注释的自定义反序列化器检测类。

应该反序列化的类

@Data
@AllArgsConstructor
@SuperBuilder
@JsonDeserialize(using = MyClassDeserializer.class)
public class MyClass{
  private final String field1;
  private final String field2;
}

应该注入的配置类

@Configuration
@ConfigurationProperties(prefix = "myconfig")
@Data
@AllArgsConstructor
@NoArgsConstructor
public class MyConfig {
   private final String confField1;
   private final String confField2;
}

MyClass 的自定义解串器:

@Component
public class MyClassDeserializer extends StdDeserializer<MyClass> {

    @Autowired
    MyConfig myConfig;

    public MyClassDeserializer () {
        this(null);
    }

    public MyClassDeserializer (Class<?> vc) {
        super(vc);
    }

    @Override
    public MyClassDeserializer deserialize(JsonParser parser, DeserializationContext context)
            throws IOException, JsonProcessingException {

         //Deserializing code that basically tries to read from myConfig
         myConfig.getConfField1(); //Hello NullPointerException my old friend
    }

}

解串器的使用

MyClass myClass = new ObjectMapper().readValue(mockJson, MyClass.class);

1 个答案:

答案 0 :(得分:3)

它不起作用的原因:

Jackson 对 Spring Boot 的东西一无所知,所以当你 readValue(..) Jackson 看到 @JsonDeserialize 带有解串器类的注解并创建解串器的新实例(它不拿起 bean,而只是 new MyClassDeserializer(..)),这就是为什么您永远不会看到 MyConfig 被注入。

如果你想让它工作,你需要通过 Spring Boot 以某种方式注册这个反序列化器,例如,像这样:How to provide a custom deserializer with Jackson and Spring Boot