Вопрос-ответ

Deserialize a List object with Gson?

Десериализовать объект List с помощью Gson?

Я хочу передать объект list через Google Gson, но я не знаю, как десериализовать универсальные типы.

Что я попробовал, просмотрев это (ответ Балуска):

MyClass mc = new Gson().fromJson(result, new List<MyClass>() {}.getClass());

но затем я получаю сообщение об ошибке в Eclipse "Тип new List<MyClass>() {} должен реализовать унаследованный абстрактный метод ...", и если я использую быстрое исправление, я получаю чудовище из более чем 20 заглушек методов.

Я почти уверен, что есть более простое решение, но, похоже, я не могу его найти!

Теперь у меня есть это:

Type listType = new TypeToken<List<MyClass>>() {}.getType();

MyClass mc = new Gson().fromJson(result, listType);

Однако я получаю следующее исключение в fromJson строке:

java.lang.NullPointerException
at org.apache.harmony.luni.lang.reflect.ListOfTypes.length(ListOfTypes.java:47)
at org.apache.harmony.luni.lang.reflect.ImplForType.toString(ImplForType.java:83)
at java.lang.StringBuilder.append(StringBuilder.java:203)
at com.google.gson.JsonDeserializerExceptionWrapper.deserialize(JsonDeserializerExceptionWrapper.java:56)
at com.google.gson.JsonDeserializationVisitor.invokeCustomDeserializer(JsonDeserializationVisitor.java:88)
at com.google.gson.JsonDeserializationVisitor.visitUsingCustomHandler(JsonDeserializationVisitor.java:76)
at com.google.gson.ObjectNavigator.accept(ObjectNavigator.java:106)
at com.google.gson.JsonDeserializationContextDefault.fromJsonArray(JsonDeserializationContextDefault.java:64)
at com.google.gson.JsonDeserializationContextDefault.deserialize(JsonDeserializationContextDefault.java:49)
at com.google.gson.Gson.fromJson(Gson.java:568)
at com.google.gson.Gson.fromJson(Gson.java:515)
at com.google.gson.Gson.fromJson(Gson.java:484)
at com.google.gson.Gson.fromJson(Gson.java:434)

Я делаю перехват JsonParseExceptions и result не равен null.

Я проверил listType с помощью отладчика и получил следующее:


  • list Type

    • args = ListOfTypes

      • list = null

      • resolvedTypes = Type[ 1 ]



    • loader = PathClassLoader

    • ownerType0 = null

    • ownerTypeRes = null

    • rawType = Class (java.util.ArrayList)

    • rawTypeName = "java.util.ArrayList"



Похоже, что getClass вызов не сработал должным образом. Есть предложения ...?

Я проверил в Руководстве пользователя Gson . Там упоминается исключение во время выполнения, которое должно произойти при преобразовании универсального типа в Json. Я сделал это "неправильно" (не показано выше), как в примере, но вообще не получил этого исключения. Поэтому я изменил сериализацию, как предложено в руководстве пользователя. Хотя это не помогло.

Редактировать:

Решено, смотрите мой ответ ниже.

Переведено автоматически
Ответ 1

Метод десериализации универсальной коллекции:

import java.lang.reflect.Type;
import com.google.gson.reflect.TypeToken;

...

Type listType = new TypeToken<ArrayList<YourClass>>(){}.getType();
List<YourClass> yourClassList = new Gson().fromJson(jsonArray, listType);

Поскольку несколько человек в комментариях упомянули об этом, вот объяснение того, как используется класс TypeToken. Конструкция new TypeToken<...>() {}.getType() фиксирует тип времени компиляции (между < и >) в объекте времени выполнения java.lang.reflect.Type. В отличие от Class объекта, который может представлять только необработанный (удаленный) тип, Type объект может представлять любой тип на языке Java, включая параметризованный экземпляр универсального типа.

У TypeToken самого класса нет открытого конструктора, потому что вы не должны создавать его напрямую. Вместо этого вы всегда создаете анонимный подкласс (отсюда {}, который является необходимой частью этого выражения).

Из-за удаления типов TypeToken класс способен фиксировать только те типы, которые полностью известны во время компиляции. (То есть вы не можете сделать new TypeToken<List<T>>() {}.getType() для параметра типа T.)

Для получения дополнительной информации см. Документацию по TypeToken классу.

Ответ 2

Другой способ - использовать массив в качестве типа, например:

MyClass[] mcArray = gson.fromJson(jsonString, MyClass[].class);

Таким образом, вы избегаете всех проблем с типом object, и если вам действительно нужен список, вы всегда можете преобразовать массив в список с помощью:

List<MyClass> mcList = Arrays.asList(mcArray);

ИМХО, это гораздо более читабельно.

И чтобы это был реальный список (который можно изменять, см. Ограничения Arrays.asList()), просто выполните следующее:

List<MyClass> mcList = new ArrayList<>(Arrays.asList(mcArray));
Ответ 3

Начиная с Gson 2.8, мы можем создать util-функцию, подобную этой:

public <T> List<T> getList(String jsonArray, Class<T> clazz) {
Type typeOfT = TypeToken.getParameterized(List.class, clazz).getType();
return new Gson().fromJson(jsonArray, typeOfT);
}

Пример использования:

String jsonArray = ...
List<User> user = getList(jsonArray, User.class);
Ответ 4

Обратитесь к этому сообщению. Тип Java Generic в качестве аргумента для GSON

У меня есть лучшее решение для этого. Вот класс-оболочка для list, чтобы оболочка могла хранить именно тот тип списка.

public class ListOfJson<T> implements ParameterizedType
{
private Class<?> wrapped;

public ListOfJson(Class<T> wrapper)
{
this.wrapped = wrapper;
}

@Override
public Type[] getActualTypeArguments()
{
return new Type[] { wrapped };
}

@Override
public Type getRawType()
{
return List.class;
}

@Override
public Type getOwnerType()
{
return null;
}
}

И тогда код может быть простым:

public static <T> List<T> toList(String json, Class<T> typeClass)
{
return sGson.fromJson(json, new ListOfJson<T>(typeClass));
}
java json generics