web-dev-qa-db-ja.com

Gsonハンドルオブジェクトまたは配列

私は以下のクラスを持っています

public class MyClass {
    private List<MyOtherClass> others;
}

public class MyOtherClass {
    private String name;
}

そして、私はこのようなJSONを持っています

{
  others: {
    name: "val"
  }
}

またはこれ

{
  others: [
    {
      name: "val"
    },
    {
      name: "val"
    }
  ]
}

これらのJSON形式の両方に同じMyClassを使用できるようにしたいと思います。 Gsonでこれを行う方法はありますか?

47
three-cups

私は答えを思いついた。

private static class MyOtherClassTypeAdapter implements JsonDeserializer<List<MyOtherClass>> {
    public List<MyOtherClass> deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext ctx) {
        List<MyOtherClass> vals = new ArrayList<MyOtherClass>();
        if (json.isJsonArray()) {
            for (JsonElement e : json.getAsJsonArray()) {
                vals.add((MyOtherClass) ctx.deserialize(e, MyOtherClass.class));
            }
        } else if (json.isJsonObject()) {
            vals.add((MyOtherClass) ctx.deserialize(json, MyOtherClass.class));
        } else {
            throw new RuntimeException("Unexpected JSON type: " + json.getClass());
        }
        return vals;
    }
}

このようにGsonオブジェクトをインスタンス化する

Type myOtherClassListType = new TypeToken<List<MyOtherClass>>() {}.getType();

Gson gson = new GsonBuilder()
        .registerTypeAdapter(myOtherClassListType, new MyOtherClassTypeAdapter())
        .create();

TypeTokencom.google.gson.reflect.TypeToken

ここでソリューションについて読むことができます:

https://sites.google.com/site/gson/gson-user-guide#TOC-Serializing-and-Deserializing-Gener

76
three-cups

ありがとうthree-cups解決策を!

複数の型に必要な場合のジェネリック型と同じこと:

public class SingleElementToListDeserializer<T> implements JsonDeserializer<List<T>> {

private final Class<T> clazz;

public SingleElementToListDeserializer(Class<T> clazz) {
    this.clazz = clazz;
}

public List<T> deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) throws JsonParseException {
    List<T> resultList = new ArrayList<>();
    if (json.isJsonArray()) {
        for (JsonElement e : json.getAsJsonArray()) {
            resultList.add(context.<T>deserialize(e, clazz));
        }
    } else if (json.isJsonObject()) {
        resultList.add(context.<T>deserialize(json, clazz));
    } else {
        throw new RuntimeException("Unexpected JSON type: " + json.getClass());
    }
    return resultList;
    }
}

そしてGsonを設定します:

Type myOtherClassListType = new TypeToken<List<MyOtherClass>>() {}.getType();
SingleElementToListDeserializer<MyOtherClass> adapter = new SingleElementToListDeserializer<>(MyOtherClass.class);
Gson gson = new GsonBuilder()
    .registerTypeAdapter(myOtherClassListType, adapter)
    .create();
8
levavare

3カップの答えを構築すると、JsonArrayを配列として直接逆シリアル化できる次のようになります。

static public <T> T[] fromJsonAsArray(Gson gson, JsonElement json, Class<T> tClass, Class<T[]> tArrClass)
        throws JsonParseException {

    T[] arr;

    if(json.isJsonObject()){
        //noinspection unchecked
        arr = (T[]) Array.newInstance(tClass, 1);
        arr[0] = gson.fromJson(json, tClass);
    }else if(json.isJsonArray()){
        arr = gson.fromJson(json, tArrClass);
    }else{
        throw new RuntimeException("Unexpected JSON type: " + json.getClass());
    }

    return arr;
}

使用法:

    String response = ".......";

    JsonParser p = new JsonParser();
    JsonElement json = p.parse(response);
    Gson gson = new Gson();
    MyQuote[] quotes = GsonUtils.fromJsonAsArray(gson, json, MyQuote.class, MyQuote[].class);
0
NameSpace