Google Gson - дезаріалізувати список <class> об'єкта? (родовий тип)


441

Я хочу перенести об’єкт списку через Google Gson, але я не знаю, як дезаріалізувати загальні типи.

Що я спробував, подивившись на це (відповідь BalusC):

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

але тоді я отримую помилку у затемненні: "Введіть новий список () {} повинен реалізувати успадкований абстрактний метод ...", і якщо я скористаюся швидким виправленням, я отримаю монстр із понад 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 і «результат» не є нульовим.

Я перевірив listType з налагоджувачем і отримав наступне:

  • список Тип
    • args = ListOfTypes
      • list = null
      • resolutionTypes = Тип [1]
    • loader = PathClassLoader
    • ownerType0 = null
    • ownerTypeRes = null
    • rawType = Клас (java.util.ArrayList)
    • rawTypeName = "java.util.ArrayList"

тому здається, що виклик "getClass" не працює належним чином. Будь-які пропозиції...?

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

Edit3: Вирішено, див. Мою відповідь нижче.


1
Відповідь, на яку ви вказували, використовує TokenType. Ви пробували так?
Нішант

просто отримав такий самий натяк, що і відповідь. наступного разу я наведу приклад детальніше. ;)
медуза

Чи можете ви спробувати реалізувати список у маркері типу? Оскільки ваш необроблений тип - це список масивів, ви повинні спробувати список масивів.
uncaught_exceptions

Відповіді:


954

Спосіб десеріалізації загальної колекції:

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класу .


31
У нових версіях GSON конвектор TypeToken не є загальнодоступним, отже, тут ви не бачите помилки конструктора. Що ви повинні зробити в цьому випадку?
Пабло

8
Використовуючи фактичну версію GSON (2.2.4), вона працює знову. Ви можете отримати доступ до конструктора тут.

мій об’єкт json починається з об'єкта, потім містить масив потрібного мені об’єкта. Я { "myObjectArray":[ {....} , {....} , {....} ] }створив файл моделі для того {....}, як мені отримати цей загальний код колекції, щоб не вважати, що мій кореневий елемент є масивом без створення нового вкладеного файлу об'єкта
CQM

7
Після імпорту потрібно --- імпорт java.lang.reflect.Type; import com.google.gson.reflect.TypeToken
Umair Saleem

4
Це добре, якщо YourClass фіксований у коді. Що робити, якщо клас приходить під час виконання?
jasxir

273

Інший спосіб - використовувати масив як тип, наприклад:

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

Таким чином ви уникнете всіх клопотів з об’єктом Type, і якщо вам справді потрібен список, ви завжди можете перетворити масив у список за допомогою:

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

ІМХО це набагато читабельніше.

А щоб це був фактичний список (який можна змінити, див. Обмеження Arrays.asList()), просто виконайте наступне:

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

4
це чудово! Як я можу використовувати його при відображенні? Я не знаю MyClassзначення, і це буде визначено динамічно!
Амін Ш.

1
nota: з цим будьте уважні, що mcList не є повноцінним списком. багато речей не вийде.
njzk2

4
Як використовувати його з дженериками? T[] yourClassList = gson.fromJson(message, T[].class);// не вдається вибрати з змінної типу
Pawel Cioch

2
@MateusViccari під час цього коментаря mcListу цій відповіді був лише результатом заклику до Arrays.asList. Цей метод повертає список, в якому більшість, якщо не всі необов'язкові методи не залишаються без змін та викидають винятки. Наприклад, ви не можете додати до цього списку жоден елемент. Як випливає з подальшого редагування, Arrays.asListобмеження мають обмеження, а перетворення їх у фактичне ArrayListдозволяє отримати список, який є кориснішим у багатьох випадках.
njzk2

2
Якщо вам потрібно побудувати тип масиву під час виконання для довільного типу елементів, ви можете використовувати, Array.newInstance(clazz, 0).getClass()як описано у відповіді Девіда Вуда .
Даніель Приден

31

Зверніться до цієї публікації. Загальний тип Java як аргумент для 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));
}

Що таке mEntity.rulePattern?
Аль-Лелопат

Це просто зразок об'єкта для тесту. Вам не потрібно про це дбати. Використовуйте метод toList і все йде добре.
Happier

@Happier Я намагаюся реалізувати цей Gson 2.8.2, і, здається, він не працює. Будь-який шанс stackoverflow.com/questions/50743932/… ви можете поглянути і повідомити мені, чого мені не вистачає
Praveen

1
@Praveen Я намагався таким чином у 2.8.2, він працює як оригінальний.
Happier

31

Оскільки 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);

2
TypeToken#getParameterizedвиглядає набагато краще, ніж хак з анонімним підкласом
Микола Кулаченко

це має бути прийнята відповідь; принаймні для новіших версій
ccpizza

2
Це ідеальна відповідь. Вирішує мою проблему.
donmj

Я скопіював ваш метод "як є", і він не працює: компілятор каже "Метод getParameterized (Class <List>, Class <T>) не визначений для типу TypeToken". Я перевірив і мою версію Gson (2.8.0), і документацію, і з цього боку все добре ... Я в кінцевому підсумку використовував рішення @Happier, яке добре працює
leguminator

@leguminator чи правильно ви імпортували TypeToken? і ви використовуєте java або kotlin. Я спробую перевірити ще раз
Фан Ван Лін

26

Веп, ще один спосіб досягти того ж результату. Ми використовуємо це для його читабельності.

Замість того, щоб робити це важко прочитане речення:

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);

9

Для Котліна просто:

import java.lang.reflect.Type
import com.google.gson.reflect.TypeToken
...
val type = object : TypeToken<List<T>>() {}.type

або, ось корисна функція:

fun <T> typeOfList(): Type {
    return object : TypeToken<List<T>>() {}.type
}

Потім, щоб використовувати:

val type = typeOfList<YourMagicObject>()

Я використовував ваш код, щоб створити цю функцію, використовуючи вдосконалені типи: inline fun <reified T> buildType() = object : TypeToken<T>() {}.type!!і зателефонуйте їй за типом списку:buildType<List<YourMagicObject>>()
coffeemakr

@coffeemakr Тут вам не потрібні повторні типи.
Чад Бінгем

Ой. Але чому ви створюєте маркер типу ArrayList у, buildTypeа також викликаєте функцію загальним типом? Це друкарня? - Це дозволило б створити ArrayList <ArrayList <YourMagicObject >>
coffeemakr

@coffeemakr ах, так. Typo
Чад Бінгем

7
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, "[{...}]");

Хороший. Але це дублює DevNGвідповідь вище, написану на 2 роки раніше: stackoverflow.com/a/17300003/1339923 (і прочитайте цю відповідь на застереження щодо цього підходу)
Ламбарт

6

Оскільки він відповідає на моє первісне запитання, я прийняв відповідь doc_180, але якщо хтось знову зіткнеться з цією проблемою, я також відповім на другу половину запитання:

NullPointerError, який я описав, не має нічого спільного з самим Списком, а з його змістом!

Клас "MyClass" не мав конструктора "no args", і не мав свого суперкласу. Після того, як я додав простий конструктор "MyClass ()" до MyClass та його надкласу, все спрацювало нормально, включаючи серіалізацію та десеріалізацію списку, як запропонував doc_180.


1
Якщо у вас є список абстрактних класів, ви отримаєте ту ж помилку. Я думаю, що це загальне повідомлення про помилку GSON для "Неможливо встановити клас".
Дрю

Порада щодо додавання конструктора допомогла мені зрозуміти, чому у мене всі нульові значення. У моєму JSON-рядку у мене були назви полів на кшталт "До" та "Від", але відповідні поля в моєму об'єкті малими значеннями були "до" та "від", тому вони були пропущені
Руна,

4

Ось рішення, яке працює з динамічно визначеним типом. Трюк - створення належного типу масиву за допомогою Array.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; 
}

Ця відповідь буде кращою, якби ви class.cast()уникали неперевіреного попередження, викликаного кастингом на (T). Або ще краще, не переймайтесь кастингом і використовуйте щось на зразок Arrays.asList()перетворення з масиву в а List<T>. Крім того, не потрібно передавати довжину до Array.newInstance()- масив розміру нуля буде досить хорошим, щоб зателефонувати getClass().
Даніель Приден

Дякую за пропозицію, я вніс запропоновані вами зміни та оновив публікацію вище.
Девід Вуд

2

Я хочу додати ще одну можливість. Якщо ви не хочете використовувати 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.


1

Зверніться до прикладу 2 для розуміння класу "Тип" Gson.

Приклад 1: У цьому десерилізованомуResturant ми використовували масив 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());
    }
}

0

Мені сподобалась відповідь від 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);

Звичайно, це не може працювати, оскільки ви намагаєтесь використовувати T у компіляційному часі. Це ефективно деріаріалізується до списку StringMap, ні?
JHH

0

У моєму випадку відповідь @ 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);
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.