web-dev-qa-db-ja.com

ジャクソンは実際にjsonをジェネリック型に逆シリアル化することはできませんか?

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

デシリアライズ-ジェネリックタイプ-with-ジャックソン

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

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

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であると推測しますが、型消去のために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)
43
lisak

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アノテーションがない場合、ジャクソンは、コンストラクターの最初の引数がresultプロパティにマップされることを知りません。

53
cambecc

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

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

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

6
yaswanth

逆シリアル化する必要があるJSON文字列には、パラメータTに関する型情報を含める必要があります。
パラメーターTにパラメーターAgentResponseとして渡すことができるすべてのクラスにJackson注釈を付ける必要があるため、パラメータータイプ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として渡すことができる各クラス(または共通のスーパータイプ)に注釈が付けられると、JacksonはパラメーターTに関する情報をJSONに含めます。このようなJSONは、コンパイル時にパラメーターTを知らなくても逆シリアル化できます。
この Jacksonのドキュメントリンク は、多態性逆シリアル化について説明していますが、この質問についても参照すると便利です。

1
Sanjeev Sachdev

プログラムで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>>>は正しく逆シリアル化されます。

0
Niclas Hedhman
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);
0
jily