Десериализовать объект 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));
}