自定义序列化程序没有默认(无 arg)构造函数

我正在使用 RestTemplate 的 REST Api。我从 API 获得的响应包含大量嵌套对象。下面是一个小片段作为示例:

"formularios": [
  {
    "form_data_id": "123006",
    "form_data": {
      "form_data_id": "123006",
      "form_id": "111",
      "efs": {
        "1": {},
        "2": "{\"t\":\"c\",\"st\":\"m\",\"v\":[{\"id\":\"3675\",\"l\":\"a) Just an example\",\"v\":\"1\"},{\"id\":\"3676\",\"l\":\"b) Another example.\",\"v\":\"2\"}]}"
      }
    }

我遇到的问题是,大多数时候“1”实际上有内容,就像“2”一样,杰克逊只是将其解析为对象“efs”上的字符串。但有时,就像在代码片段中一样,API将其发送到空,jackson将其作为对象,这给了我一个错误,说了一些关于START_OBJECT(不记得确切的错误,但这对这个问题并不重要)。

所以我决定做一个自定义的反序列化程序,这样当jackson读取“1”时,它会忽略空对象,只将其解析为空字符串。

这是我的自定义反序列化程序:

public class CustomDeserializer extends StdDeserializer<Efs> {

 public CustomDeserializer(Class<Efs> t) {
     super(t);
 }

 @Override
 public Efs deserialize(JsonParser jp, DeserializationContext dc)
         throws IOException, JsonProcessingException {

     String string1 = null;
     String string2 = null;
     JsonToken currentToken = null;

     while ((currentToken = jp.nextValue()) != null) {
         if (currentToken.equals(JsonToken.VALUE_STRING)) {
             if (jp.getCurrentName().equals("1")) {
                 string1 = jp.getValueAsString();
             } else {
                 string2 = jp.getValueAsString();
             }

         } else {
             if (jp.getCurrentName().equals("2")) {
                 string2 = jp.getValueAsString();
             }

         }
     }
     return new Efs(string1, string2);

  }
 }

这是我在接收来自 API 的响应时使用它的方式:

    ObjectMapper mapper = new ObjectMapper();  
    SimpleModule mod = new SimpleModule("EfsModule");
    mod.addDeserializer(Efs.class, new CustomDeserializer(Efs.class));
    mapper.registerModule(mod);


    List<HttpMessageConverter<?>> messageConverters = new ArrayList<>();
    MappingJackson2HttpMessageConverter jsonMessageConverter = new MappingJackson2HttpMessageConverter();
    jsonMessageConverter.setObjectMapper(mapper);
    messageConverters.add(jsonMessageConverter);


    RestTemplate restTemplate = new RestTemplate();
    restTemplate.setMessageConverters(messageConverters);

我收到错误:

 CustomDeserializer has no default (no arg) constructor

但我不知道我到底做错了什么,也不知道该如何解决。感谢您的帮助和对这个长问题的歉意,我想尽可能多地提供背景信息。


答案 1

用户还可以陷入一个陷阱(比如我自己)。如果将反序列化程序声明为内部类(而不是静态嵌套类),如下所示:

@JsonDeserialize(using = DomainObjectDeserializer.class)
public class DomainObject {
    private String key;

    public class DomainObjectDeserializer extends StdDeserializer<DomainObject> {
        public DomainObjectDeserializer() {
            super(DomainObject.class);
        }

        @Override
        public DomainObject deserialize(JsonParser p, DeserializationContext ctxt) throws IOException {
            // code
        }
    }
}

Jackson 使用 Class#getDeclaredConstructor() 而不带参数(方法接受),这意味着:给我一个默认的(无参数)构造函数。当 Jackson 尝试创建时,上面的代码将引发异常,因为生成的构造函数接受封闭的类引用。从技术上讲,没有默认构造函数。varargDomainObjectDeserializerjavacDomainObjectDeserializer

出于好奇,您可以执行并确保该方法返回包含构造函数定义的单元素数组,并包含封闭的类引用。DomainObjectDeserializer.class.getDeclaredConstructors()

应将其声明为类。DomainObjectDeserializerstatic

这里有一个很好的答案,可以更详细地阅读。


答案 2

需要有一个不带参数的默认构造函数。您可以做的是创建一个(或者如果您真的不需要它,请替换另一个):

public class CustomDeserializer extends StdDeserializer<Efs> {

   public CustomDeserializer() {
       super(Efs.class);
   }
   ...
}

推荐