46

次の質問が面倒であるか、まったく回答されていないため、これは重複した質問です。

ジャクソンを使用したジェネリック型の逆シリアル化

jackson-deserialize-to-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}}

およびジェネリック型を逆シリアル化する 2 つの推奨される方法:

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)
4

6 に答える 6

8

同じアプローチを使用してみましたが、モデル クラスに注釈を付けていません。それは私にとってはうまくいきました。

これは私のモデルクラスです

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 が含まれます。

于 2016-08-18T11:37:25.657 に答える