Подтвердить что ты не робот

Действительно ли Джексон не может десериализовать json в родовой тип?

Это повторяющийся вопрос, потому что следующие вопросы либо беспорядочны, либо вообще не отвечают:

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, но находит аргумент конструктора типа объекта из-за стирания типа и выдает ошибку. Так это ошибка Джексона, или я делаю что-то неправильно? Что еще есть явная спецификация 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: [email protected]; 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)
4b9b3361

Ответ 1

Вам нужно добавить некоторые аннотации к конструктору, чтобы сообщить Джексону, как построить объект. Для меня работали следующие:

public class AgentResponse<T> {

    private T result;

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

Без аннотации @JsonCreator, Джексон не может знать, как вызвать этот конструктор. И без аннотации @JsonProperty, Джексон не знает, что первый аргумент конструктора сопоставляется с свойством result.

Ответ 2

Я попытался использовать тот же подход, но я не аннотировал мой класс модели. Это сработало для меня.

Это мой модельный класс

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 в строке.

Ответ 3

Строка JSON, которую необходимо десериализовать, должна содержать информацию о типе параметра T.
Вам нужно будет поместить примечания Джексона в каждый класс, который можно передать как параметр T в класс AgentResponse, чтобы информация о типе параметра типа T могла быть прочитана/записана в строку JSON Джексоном.

Предположим, что 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, аннотирован, Джексон включит информацию о параметре T в JSON. Такой JSON можно затем десериализовать, не зная параметра T во время компиляции.
Эта ссылка на документацию Джексона говорит о полиморфной десериализации, но полезно также обратиться к этому вопросу.

Ответ 4

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

Если вы программно выбираете 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>>> будет десериализован правильно.