这是一个重复的问题,因为以下问题要么是凌乱,要么根本没有回答:
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);
杰克逊永远无法处理泛型类型T,它认为它是来自JavaType的Map,但是由于类型擦除它会找到对象类型构造函数参数并抛出错误。这是杰克逊的错误,还是我做错了什么?还有什么是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)
答案 0 :(得分:53)
您需要在构造函数上添加一些注释,以告诉Jackson如何构建对象。以下对我有用:
public class AgentResponse<T> {
private T result;
@JsonCreator
public AgentResponse(@JsonProperty("result") T result) {
this.result = result;
}
public T getResult() {
return result;
}
}
没有@JsonCreator
注释,杰克逊无法知道调用此构造函数。如果没有@JsonProperty
注释,Jackson就不知道构造函数的第一个参数映射到result
属性。
答案 1 :(得分:6)
我尝试使用相同的方法,但我还没有注释我的模型类。它对我来说很好。
这是我的模特课
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。
答案 2 :(得分:1)
需要反序列化的JSON字符串必须包含有关参数T
的类型信息。
您必须将Jackson注释放在可以作为参数T
传递给类AgentResponse
的每个类上,以便可以从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讨论了多态反序列化,但也可用于引用这个问题。
答案 3 :(得分:1)
如果您以编程方式从方法返回类型或字段中选取java.lang.reflect.Type
,则最容易使用
Type type = ...;
ObjectMapper mapper = new ObjectMapper();
JavaType javaType = mapper.getTypeFactory().constructType( type );
Object value = mapper.readValue( json, javaType );
已创建完全嵌套的JavaType,因此Controller<PID<Temperature,Double>>>
将被正确地反序列化。
答案 4 :(得分:0)
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);
答案 5 :(得分:0)
解决方案:
使用类型引用代替类
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>>() {});