Google Gson - deserialize list <class> объект? (общий тип)
Я хочу передать объект списка через 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)
Я улавливаю JsonParseExceptions и "result" не является нулевым.
Я проверил listType с помощью отладчика и получил следующее:
- список Тип
- args = ListOfTypes
- list = null
- resolvedTypes = Тип [1]
- loader = PathClassLoader
- ownerType0 = null
- ownerTypeRes = null
- rawType = класс (java.util.ArrayList)
- rawTypeName = "java.util.ArrayList"
поэтому кажется, что вызов getClass не работает должным образом. Любые предложения...?
Edit2:
Я проверил в Руководство пользователя Gson. В нем упоминается исключение Runtime Exception, которое должно произойти при разборе родового типа Json. Я сделал это "неправильно" (не показано выше), как и в примере, но не получил этого исключения вообще. Поэтому я изменил сериализацию, как в предлагаемом руководстве пользователя. Однако не помогло.
Edit3:
Решенный, см. Мой ответ ниже.
Ответы
Ответ 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);
Ответ 2
Другой способ - использовать массив как тип, например:
MyClass[] mcArray = gson.fromJson(jsonString, MyClass[].class);
Таким образом, вы избегаете всех проблем с объектом Type, и если вам действительно нужен список, вы всегда можете преобразовать массив в список:
List<MyClass> mcList = Arrays.asList(mcArray);
ИМХО это гораздо читаемо.
И чтобы сделать его реальным списком (который можно изменить, см. ограничения Arrays.asList()
), просто выполните следующее:
List<MyClass> mcList = new ArrayList<>(Arrays.asList(mcArray));
Ответ 3
Для тех, кто получает неоднозначные предложения импорта текста, это тот, который вы хотите.
import java.lang.reflect.Type;
Ответ 4
Обратитесь к этому сообщению.
Java Type 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));
}
Ответ 5
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);
Ответ 6
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, "[{...}]");
Ответ 7
Как он отвечает на мой первоначальный вопрос, я принял ответ doc_180, но если кто-то снова столкнется с этой проблемой, я также отвечу на вторую половину моего вопроса:
Описанный мной NullPointerError не имеет ничего общего с самим списком, но с его содержимым!
В классе "MyClass" не было конструктора "no args", и у него не было своего суперкласса. Как только я добавил простой конструктор MyClass() в MyClass и его суперкласс, все сработало нормально, включая сериализацию List и десериализацию, как это было предложено doc_180.
Ответ 8
Вот решение, которое работает с динамически определенным типом. Трюк создает правильный тип массива с помощью Array.newInstance().
public static <T> List<T> fromJsonList(String json, Class<T> clazz) {
Object [] array = (Object[])java.lang.reflect.Array.newInstance(clazz, 1);
array = gson.fromJson(json, array.getClass());
List<T> list = new ArrayList<T>();
for (int i=0 ; i<array.length ; i++)
list.add((T)array[i]);
return list;
}
Ответ 9
Я хочу добавить еще одну возможность. Если вы не хотите использовать 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();
Теперь вы можете получить доступ к каждому элементу объекта className.
Ответ 10
Для Котлина просто:
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>()
Ответ 11
Мне понравился ответ от 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);
Ответ 12
В моем случае ответ @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);
Ответ 13
Обратитесь к примеру 2 для понимания класса G Type.
Пример 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());
}
}