Использование GSON для разбора массива с несколькими типами - PullRequest
12 голосов
/ 21 марта 2011

Я хочу использовать GSON для анализа следующего json:

[
    [
        "hello",
        1,
        [2]
    ],
    [
        "world",
        3,
        [2]
    ]
]

Итак, это 1 массив, содержащий 2 массива.2 внутренних массива сами по себе являются массивами, состоящими из типов String, int, array.

Я не уверен, как я могу использовать классы Java для моделирования массива, который имеет 3 различных типа (String, int, array).Я начинаю с:

// String json just contains the aforementioned json string.

ArrayList<ArrayList<XXX>> data = new ArrayList<ArrayList<XXX>>();

Type arrayListType = new TypeToken<ArrayList<ArrayList<XXX>>>(){}.getType();

data = gson.fromJson(json, arrayListType);

Но что должно быть там, где «ХХХ»?Я думаю, что это должен быть массив, но это должен быть массив с 3 различными типами данных.Итак, как я могу использовать Java для моделирования этого?

Может ли помочь?Спасибо.

Ответы [ 2 ]

14 голосов
/ 02 июня 2011

Gson имеет специальную обработку для десериализации некоторых однокомпонентных массивов в не-массив типа. Например, int data = gson.fromJson("[3]", int.class); присваивает данным значение int 3.

Конечно, десериализация однокомпонентного массива в не-массив типа не требуется. Например, предыдущий пример может быть десериализован как int[] data = gson.fromJson("[3]", int[].class);.

Gson также часто десериализует не строковое значение в строку, когда его спрашивают. Применяя это к первому примеру, String data = gson.fromJson("[3]", String.class); работает так же хорошо.

Обратите внимание, что Gson не работает с десериализацией первого примера как типа Object. Object data = gson.fromJson("[3]", Object.class); приводит к исключению синтаксического анализа с жалобой на то, что [3] не является примитивом.

Применительно к примеру из исходного вопроса выше, если допустимо обрабатывать все значения как строки, тогда десериализация становится простой.

// output:
// hello 1 2 
// world 3 2 

public class Foo
{
  static String jsonInput = 
    "[" +
      "[\"hello\",1,[2]]," +
      "[\"world\",3,[2]]" +
    "]";

  public static void main(String[] args)
  {
    Gson gson = new Gson();
    String[][] data = gson.fromJson(jsonInput, String[][].class);
    for (String[] data2 : data)
    {
      for (String data3 : data2)
      {
        System.out.print(data3);
        System.out.print(" ");
      }
      System.out.println();
    }
  }
}

К сожалению, с Gson я не смог найти простой подход десериализации, который позволял бы «лучше» связывать более специфичные и смешанные типы в массиве, поскольку Java не предоставляет синтаксиса для определения смешанного Тип массива. Например, предпочтительный тип коллекции в исходном вопросе может быть List<List<String, int, List<int>>>, но это невозможно определить в Java. Итак, вы должны довольствоваться List<List<String>> (or String[][]) или перейти к подходу с более «ручным» анализом.

(Да, Java допускает объявление типа List<List<Object>>, но Object не является достаточно конкретным типом для значимой десериализации. Также, как обсуждалось, попытка десериализации [3] в Object приводит к исключению синтаксического анализа. )


Небольшое обновление: мне недавно пришлось десериализовать какой-то небрежный JSON, включающий структуру, не слишком отличающуюся от таковой в первоначальном вопросе. В итоге я использовал собственный десериализатор для создания объекта из грязного массива JSON. Аналогично следующему примеру.

// output: 
// [{MyThreeThings: first=hello, second=1, third=[2]}, 
//  {MyThreeThings: first=world, second=3, third=[4, 5]}]

import java.lang.reflect.Type;
import java.util.Arrays;

import com.google.gson.Gson;
import com.google.gson.GsonBuilder;
import com.google.gson.JsonArray;
import com.google.gson.JsonDeserializationContext;
import com.google.gson.JsonDeserializer;
import com.google.gson.JsonElement;
import com.google.gson.JsonParseException;

public class FooToo
{
  static String jsonInput =
      "[" +
          "[\"hello\",1,[2]]," +
          "[\"world\",3,[4,5]]" +
      "]";

  public static void main(String[] args)
  {
    GsonBuilder gsonBuilder = new GsonBuilder();
    gsonBuilder.registerTypeAdapter(MyThreeThings.class, new MyThreeThingsDeserializer());
    Gson gson = gsonBuilder.create();
    MyThreeThings[] things = gson.fromJson(jsonInput, MyThreeThings[].class);
    System.out.println(Arrays.toString(things));
  }
}

class MyThreeThings
{
  String first;
  int second;
  int[] third;

  MyThreeThings(String first, int second, int[] third)
  {
    this.first = first;
    this.second = second;
    this.third = third;
  }

  @Override
  public String toString()
  {
    return String.format(
        "{MyThreeThings: first=%s, second=%d, third=%s}",
        first, second, Arrays.toString(third));
  }
}

class MyThreeThingsDeserializer implements JsonDeserializer<MyThreeThings>
{
  @Override
  public MyThreeThings deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context)
      throws JsonParseException
  {
    JsonArray jsonArray = json.getAsJsonArray();
    String first = jsonArray.get(0).getAsString();
    int second = jsonArray.get(1).getAsInt();
    JsonArray jsonArray2 = jsonArray.get(2).getAsJsonArray();
    int length = jsonArray2.size();
    int[] third = new int[length];
    for (int i = 0; i < length; i++)
    {
      int n = jsonArray2.get(i).getAsInt();
      third[i] = n;
    }
    return new MyThreeThings(first, second, third);
  }
}

Руководство пользователя Gson охватывает обработку десериализации коллекций смешанных типов на примере, аналогичном приведенному в в разделе «Сериализация и десериализация коллекции с объектами произвольных типов» .

0 голосов
/ 21 марта 2011

Во-первых, я думаю, что вы можете ошибаться в приведенном выше примере. Массив, состоящий из трех разных, - это очень необычный подход, если не сказать больше. Вероятно, ваша структура json - это массив, содержащий кортежи. Эти кортежи затем включают в себя массив.

Как:

[
{
    "hello",
    1,
    [2]
},
{
    "world",
    3,
    [2]
}
]

XXX должен быть объектом, содержащим:

Строка

Int (или Integer)

Массив (я полагаю) целых.

Затем вы создаете массив из этих объектов и анализируете в него json.

Тем не менее, ваш JSON кажется действительно плохо сформированным, так как все члены должны быть названы, как

[
{
    "str":"hello",
    "intVal":1,
    "intArr":[2]
},
{
    "str":"world",
    "intVal":3,
    "intArr":[2]
}
]

Если, с другой стороны, JSON действительно выглядит так, как вы его описываете, вам придется создавать массивы Object, простые и понятные, а затем приводить их при чтении из структуры данных.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...