Google Gson-десериализовать объект списка? (универсальный тип)
Я хочу передать объект списка через Google Gson, но я не знаю, как десериализовать общие типы.
то, что я пытался, глядя на эту (ответ BalusC это):
MyClass mc = new Gson().fromJson(result, new List<MyClass>(){}.getClass());
но затем я получаю ошибку в eclipse, говоря: "тип new List(){} должен реализовать унаследованный абстрактный метод..."и если я использую быстрое исправление, я получаю монстра из более чем 20 окурков метода.
Я уверен, что есть более простое решение, но я не могу найти это!
Edit:
Теперь у меня есть
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)
Я do catch JsonParseExceptions и "результат" не является нулем.
Я проверил listType с отладчиком и получил следующее:
- тип списка
- args = ListOfTypes
- list = null
- resolvedTypes = Type[ 1 ]
- loader = PathClassLoader
- ownerType0 = null
- ownerTypeRes = null
- rawType = Class (java.утиль.ArrayList)
- rawTypeName = " java.утиль.ArrayList"
- args = ListOfTypes
Так что, похоже, вызов "getClass" не работал должным образом. Любое предложение...?
Edit2:
Я проверил на Руководство Пользователя Gson. В нем упоминается исключение времени выполнения, которое должно происходит во время разбора универсального типа в Json. Я сделал это "неправильно" (не показано выше), как и в Примере, но не получил этого исключения вообще. Поэтому я изменил сериализацию, как в руководстве пользователя предложил. Но это не помогло.
Edit3:
Решено, см. Мой ответ ниже.
13 ответов:
метод десериализации универсальной коллекции:
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()захватывает тип времени компиляции (между<и>) во время работы
другой способ-использовать массив в качестве типа, например:
MyClass[] mcArray = gson.fromJson(jsonString, MyClass[].class);таким образом, Вы избежать всех хлопот, связанных с объектом типа, и если вам действительно нужен список, вы всегда можете преобразовать массив в список:
List<MyClass> mcList = Arrays.asList(mcArray);ИМХО это гораздо более читабельным.
и сделать его фактическим списком (который может быть изменен, см. Ограничения
Arrays.asList()), то просто сделайте следующее:List<MyClass> mcList = new ArrayList<>(Arrays.asList(mcArray));
см. этот пост. тип Java Generic в качестве аргумента для GSON
у меня есть лучшее решение для этого. Вот класс-оболочка для списка, поэтому оболочка может хранить именно тип списка.
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)); }
Wep, другой способ достичь того же результата. Мы используем его для чтения.
вместо того, чтобы делать это трудночитаемое предложение:
Type listType = new TypeToken<ArrayList<YourClass>>(){}.getType(); List<YourClass> list = new Gson().fromJson(jsonArray, listType);создать пустой класс, который расширяет список вашего объекта:
public class YourClassList extends ArrayList<YourClass> {}и использовать его при разборе JSON:
List<YourClass> list = new Gson().fromJson(jsonArray, YourClassList.class);
public static final <T> List<T> getList(final Class<T[]> clazz, final String json) { final T[] jsonToObject = new Gson().fromJson(json, clazz); return Arrays.asList(jsonToObject); }пример:
getList(MyClass[].class, "[{...}]");
поскольку он отвечает на мой первоначальный вопрос, я принял ответ doc_180, но если кто-то снова столкнется с этой проблемой, я также отвечу на вторую половину моего вопроса:
nullpointererror, который я описал, не имел ничего общего с самим списком, но с его содержанием!
класс "MyClass "не имел конструктора" no args", и у него не было своего суперкласса. Как только я добавил простой конструктор "MyClass ()" в MyClass и его суперкласс, все сработало отлично, включая сериализацию списка и десериализацию, как предложено doc_180.
вот решение, которое работает с динамически определяемым типом. Хитрость заключается в создании правильного типа массива с помощью массива.newInstance ().
public static <T> List<T> fromJsonList(String json, Class<T> clazz) { Object [] array = (Object[])java.lang.reflect.Array.newInstance(clazz, 0); array = gson.fromJson(json, array.getClass()); List<T> list = new ArrayList<T>(); for (int i=0 ; i<array.length ; i++) list.add(clazz.cast(array[i])); return list; }
для Котлина просто:
import java.lang.reflect.Type import com.google.gson.reflect.TypeToken ... val type = object : TypeToken<ArrayList<T>>() {}.typeили, вот полезная функция:
fun <T> buildType(): Type { return object : TypeToken<ArrayList<T>>() {}.type }затем, чтобы использовать:
val type = buildType<YourMagicObject>()
Я хочу добавить еще одну возможность. Если вы не хотите использовать TypeToken и хотите преобразовать массив объектов json в ArrayList, то вы можете поступить следующим образом:
Если ваша структура json похожа:
{ "results": [ { "a": 100, "b": "value1", "c": true }, { "a": 200, "b": "value2", "c": false }, { "a": 300, "b": "value3", "c": true } ]}
и ваша структура класса выглядит так:
public class ClassName implements Parcelable { public ArrayList<InnerClassName> results = new ArrayList<InnerClassName>(); public static class InnerClassName { int a; String b; boolean c; } }затем вы можете разобрать его как:
Gson gson = new Gson(); final ClassName className = gson.fromJson(data, ClassName.class); int currentTotal = className.results.size();Теперь вы можете получить доступ к каждому элементу объекта класса.
С
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);
мне понравился ответ от kays1, но я не мог его реализовать. Поэтому я построил свою собственную версию, используя его концепцию.
public class JsonListHelper{ public static final <T> List<T> getList(String json) throws Exception { Gson gson = new GsonBuilder().setDateFormat("yyyy-MM-dd HH:mm:ss").create(); Type typeOfList = new TypeToken<List<T>>(){}.getType(); return gson.fromJson(json, typeOfList); } }использование:
List<MyClass> MyList= JsonListHelper.getList(jsonArrayString);
в моем случае ответ @uncaught_exceptions не сработал, мне пришлось использовать
List.classвместоjava.lang.reflect.Type:String jsonDuplicatedItems = request.getSession().getAttribute("jsonDuplicatedItems").toString(); List<Map.Entry<Product, Integer>> entries = gson.fromJson(jsonDuplicatedItems, List.class);
см. Пример 2 для понимания класса "Type" Gson.
Пример 1: в этом deserilizeResturant мы использовали Employee [] массив и получить детали
public static void deserializeResturant(){ String empList ="[{\"name\":\"Ram\",\"empId\":1},{\"name\":\"Surya\",\"empId\":2},{\"name\":\"Prasants\",\"empId\":3}]"; Gson gson = new Gson(); Employee[] emp = gson.fromJson(empList, Employee[].class); int numberOfElementInJson = emp.length(); System.out.println("Total JSON Elements" + numberOfElementInJson); for(Employee e: emp){ System.out.println(e.getName()); System.out.println(e.getEmpId()); } }Пример 2:
//Above deserilizeResturant used Employee[] array but what if we need to use List<Employee> public static void deserializeResturantUsingList(){ String empList ="[{\"name\":\"Ram\",\"empId\":1},{\"name\":\"Surya\",\"empId\":2},{\"name\":\"Prasants\",\"empId\":3}]"; Gson gson = new Gson(); // Additionally we need to se the Type then only it accepts List<Employee> which we sent here empTypeList Type empTypeList = new TypeToken<ArrayList<Employee>>(){}.getType(); List<Employee> emp = gson.fromJson(empList, empTypeList); int numberOfElementInJson = emp.size(); System.out.println("Total JSON Elements" + numberOfElementInJson); for(Employee e: emp){ System.out.println(e.getName()); System.out.println(e.getEmpId()); } }
Comments