【问题标题】:Is Jackson really unable to deserialize json into a generic type?Jackson 真的无法将 json 反序列化为泛型类型吗?
【发布时间】:2013-06-28 08:39:03
【问题描述】:

这是一个重复的问题,因为以下问题要么乱七八糟,要么根本没有答案:

deserializing-a-generic-type-with-jackson

jackson-deserialize-into-runtime-specified-class

jackson-deserialize-using-generic-class

jackson-deserialize-generic-class-variable

我希望这个问题最终能找到一个让这个问题永远清晰的答案。

有一个模型:

public class AgentResponse<T> {

    private T result;

    public AgentResponse(T result) {
        this.result = result;
    }
    public T getResult() {
        return result;
    }
}

JSON 输入:

{"result":{"first-client-id":3,"test-mail-module":3,"third-client-id":3,"second-client-id":3}}

还有两种反序列化泛型类型的推荐方法:

mapper.readValue(out, new TypeReference<AgentResponse<Map<String, Integer>>>() {}); 

JavaType javaType = mapper.getTypeFactory().constructParametricType(AgentResponse.class, Map.class);
mapper.readValue(out, javaType);

Jackson 永远无法处理泛型类型 T,它认为它是来自 JavaType 的 Map,但由于类型擦除,它找到了 Object 类型的构造函数参数并抛出错误。那么这是杰克逊的错误,还是我做错了什么? TypeReference 或 JavaType 的显式规范还有什么用途?

com.fasterxml.jackson.databind.JsonMappingException: No suitable constructor found for type [simple type, class com.fg.mail.smtp.AgentResponse<java.util.Map<java.lang.String,java.lang.Integer>>]: can not instantiate from JSON object (need to add/enable type information?)
at [Source: java.io.InputStreamReader@4f2d26d; line: 1, column: 2]
at com.fasterxml.jackson.databind.JsonMappingException.from(JsonMappingException.java:164)
at com.fasterxml.jackson.databind.deser.BeanDeserializerBase.deserializeFromObjectUsingNonDefault(BeanDeserializerBase.java:984)
at com.fasterxml.jackson.databind.deser.BeanDeserializer.deserializeFromObject(BeanDeserializer.java:276)
at com.fasterxml.jackson.databind.deser.BeanDeserializer.deserialize(BeanDeserializer.java:121)
at com.fasterxml.jackson.databind.ObjectMapper._readMapAndClose(ObjectMapper.java:2888)
at com.fasterxml.jackson.databind.ObjectMapper.readValue(ObjectMapper.java:2064)

【问题讨论】:

  • 你应该在 jackson-users 邮件列表上发布这个问题
  • 另外,您是否真的尝试反序列化 all 该 JSON 或仅此 JSON 的 result 成员值?
  • 我正在反序列化 JSON 输入到 AgentResponse。我不知道如何以及为什么要部分反序列化它。它只是一个带有状态和结果值的响应,可能是对象、集合、地图等。我修改了问题,所以只有通用结果,

标签: java generics jackson deserialization


【解决方案1】:

解决方案:

使用 TypeReference 代替类

 public T getObject(String json, TypeReference typeReference) throws JsonParseException, JsonMappingException, IOException{
      ObjectMapper mapper = new ObjectMapper();
      mapper.configure(DeserializationFeature.UNWRAP_ROOT_VALUE, true);
     T t = (T) mapper.readValue(json, typeReference);
      return t;
   }

我的 Pojo 结构

    public class SQSRequest<T> implements Serializable {    
    private T data;
    private String msgId;
    private String msgGroupId;
    .... 
}


        
 public class EmailDetails implements Serializable {
     private Map<String,String> paramMap;
        .... 
} 

使用

SQSRequest<EmailDetails> req2=this.getObject(str, new TypeReference<SQSRequest<EmailDetails>>() {});

【讨论】:

    【解决方案2】:

    如果您以编程方式从例如方法返回类型或字段中获取java.lang.reflect.Type,那么它最容易使用

    Type type = ...;
    ObjectMapper mapper = new ObjectMapper();
    JavaType javaType = mapper.getTypeFactory().constructType( type );
    Object value = mapper.readValue( json, javaType );
    

    创建了一个完全嵌套的 JavaType,因此 Controller&lt;PID&lt;Temperature,Double&gt;&gt;&gt; 将被正确反序列化。

    【讨论】:

    • 请你举个例子 type type = ....;
    【解决方案3】:
    public class AgentResponse<T> {
    
    private T result;
    
    public AgentResponse(T result) {
        this.result = result;
    }
    public T getResult() {
        return result;
    }
    

    }

    所以对于上面的类结构,T可以是T1,T2类

    所以要对 AgentResponse 进行反序列化,请使用以下代码

    JavaType javaType = objectMapper.getTypeFactory.constructParametricType(AgentResponse.class,T1.class)
    AgentResponse<T1> agentResponseT1 = objectMapper.readValue(inputJson,javaType);
    

    【讨论】:

      【解决方案4】:

      需要反序列化的JSON字符串必须包含参数T的类型信息。
      您必须在每个可以作为参数T 传递给类AgentResponse 的类上放置Jackson 注释,以便Jackson 可以从/写入JSON 字符串中读取/写入有关参数类型T 的类型信息。

      让我们假设T 可以是任何扩展抽象类Result 的类。

      public class AgentResponse<T extends Result> {
          public Hits<T> hits;
      }
      
      @JsonTypeInfo(use = JsonTypeInfo.Id.NAME, include = JsonTypeInfo.As.WRAPPER_OBJECT)
      @JsonSubTypes({
              @JsonSubTypes.Type(value = ImageResult.class, name = "ImageResult"),
              @JsonSubTypes.Type(value = NewsResult.class, name = "NewsResult")})
      public abstract class Result {
      
      }
      
      public class ImageResult extends Result {
      
      }
      
      public class NewsResult extends Result {
      
      }
      

      一旦可以作为参数T 传递的每个类(或其公共超类型)都被注释,Jackson 将在 JSON 中包含有关参数T 的信息。然后可以在编译时不知道参数T 的情况下反序列化此类 JSON。
      这个Jackson documentation link 讨论了多态反序列化,但对于这个问题也很有用。

      【讨论】:

      • 知道如果类是 Integer 或任何其他内置类该怎么办?
      【解决方案5】:

      我尝试使用相同的方法,但我没有注释我的模型类。对我来说效果很好。

      这是我的模型类

      public class BasicMessage<T extends Serializable> implements Message<T> {
          private MessageHeader messageHeader = new MessageHeader();
          private T payload;
          public MessageHeader getHeaders() {
              return messageHeader;
          }
      
          public Object getHeader(String key) {
              return messageHeader.get(key);
          }
      
          public Object addHeader(String key, Object header) {
              return messageHeader.put(key, header);
          }
      
          public T getPayload() {
              return payload;
          }
      
          public void setPayload(T messageBody) {
              this.payload = messageBody;
          }
      }
      

      我使用了下面的方法来反序列化payload

      public static <T extends Serializable> BasicMessage<T> getConcreteMessageType(String jsonString, Class<T> classType) {
              try {
                  ObjectMapper mapper = new ObjectMapper();
                  JavaType javaType = mapper.getTypeFactory().constructParametricType(BasicMessage.class, classType);
                  return mapper.readValue(jsonString, javaType);
              } catch (IOException e) {
      
              }
       }
      

      其中 jsonString 包含字符串中的 BasicMessageObject。

      【讨论】:

      • +1 ...您仍然会失去创建对象的不变性。或者私人二传手也可以做到这一点?
      • 看起来我们需要为 Java 中的所有泛型类型转换。这修复了内部类型问题。
      【解决方案6】:

      你需要在构造函数上添加一些注解来告诉Jackson如何构建对象。以下对我有用:

      public class AgentResponse<T> {
      
          private T result;
      
          @JsonCreator
          public AgentResponse(@JsonProperty("result") T result) {
              this.result = result;
          }
          public T getResult() {
              return result;
          }
      }
      

      没有@JsonCreator 注解,Jackson 无法知道调用此构造函数。如果没有 @JsonProperty 注释,Jackson 不知道构造函数的第一个参数映射到 result 属性。

      【讨论】:

      • 你救了我的一天,我一直在调试杰克逊堆栈,我看到它找到了构造函数参数,所以我认为这是一般问题。事实上,真正的问题是杰克逊不知道构造函数参数名称,因为 JVM 没有提供它。该死的,我应该先尝试不使用泛型...谢谢!
      • 如果你不能或不愿意直接注释类,你可以用一个mixin来代替:wiki.fasterxml.com/JacksonMixInAnnotations
      • 顺便说一句:如果构造函数中有很多参数,则必须使用@JsonProperty 注释所有参数
      • 是否有任何特殊代码需要执行 dezerialise ?或标准杰克逊反序列化将适用于上述方法?我见过有人用类型引用编写特殊的反序列化代码,因此提出了这个问题。
      猜你喜欢
      • 1970-01-01
      • 2017-12-23
      • 2012-10-18
      • 1970-01-01
      • 2012-07-24
      • 1970-01-01
      相关资源
      最近更新 更多