杰克逊真的无法将json反序列化为泛型吗?

新手上路,请多包涵

这是一个重复的问题,因为以下问题要么很乱,要么根本没有回答:

用杰克逊反序列化通用类型

杰克逊反序列化到运行时指定的类

杰克逊反序列化使用通用类

杰克逊反序列化通用类变量

我希望这个问题最终能找到一个答案,使这一点永远清楚。

有一个模型:

 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)

原文由 lisak 发布,翻译遵循 CC BY-SA 4.0 许可协议

阅读 454
2 个回答

您需要在构造函数上添加一些注释来告诉 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 属性。

原文由 cambecc 发布,翻译遵循 CC BY-SA 3.0 许可协议

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

这是我的模型课

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;
    }
}

我使用以下方法反序列化有效负载

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。

原文由 yaswanth 发布,翻译遵循 CC BY-SA 3.0 许可协议

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题