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"




Так что, похоже, вызов "getClass" не работал должным образом. Любое предложение...?



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



Edit3:
Решено, см. Мой ответ ниже.

546   13  

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

    Ничего не найдено.