使用 GSON 解析具有多种类型的数组

2022-09-04 03:36:38

我希望使用GSON来解析以下json:

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

所以,这是1个数组,包含2个数组。2 个内部数组本身就是数组,由 String、int、数组类型组成。

我不确定如何使用Java类来建模具有3种不同类型(字符串,int,数组)的数组。我从以下几点开始:

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

但是,“XXX”应该是什么?我认为它应该是一个数组,但它应该是一个具有3种不同数据类型的数组。那么我如何使用Java来建模呢?

能提供任何帮助吗?谢谢。


答案 1

Gson 具有将某些单组件数组反序列化为非数组类型的特殊处理。例如,将 int 值 3 分配给数据。int data = gson.fromJson("[3]", int.class);

当然,不需要将单组件数组反序列化为非数组类型。例如,前面的示例可以反序列化为 。int[] data = gson.fromJson("[3]", int[].class);

当被问到时,Gson 还经常会将非 String 值反序列化为 String。将此应用于第一个示例,效果也同样有效。String data = gson.fromJson("[3]", String.class);

请注意,告诉 Gson 将第一个示例反序列化为 Object 类型是行不通的。 导致解析异常,抱怨 [3] 不是基元。Object data = gson.fromJson("[3]", Object.class);

应用于上面原始问题中的示例,如果将所有值视为字符串是可以接受的,则反序列化变得简单。

// 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不提供用于定义混合类型数组的语法。例如,原始问题中集合的首选类型可能是 ,但这无法在 Java 中定义。因此,您必须满足于 ,或者转向具有更多“手动”解析的方法。List<List<String, int, List<int>>>List<List<String>> (or String[][])

(是的,Java 允许类型声明 ,但不是足够具体的类型,无法有意义地反序列化为。此外,如前所述,尝试将 [3] 反序列化为 Object 会导致解析异常。List<List<Object>>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 用户指南介绍了如何处理混合类型集合的反序列化,并在“使用任意类型的对象序列化和反序列化集合”一节中提供了类似的示例。


答案 2

首先,我认为你在上面的例子中可能是错误的。至少可以说,由三个不同组成的数组是一种非常不寻常的方法。您的 json 结构可能是一个数组,包含元组。然后,这些元组包含一个数组。

喜欢:

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

XXX 应该是一个包含以下内容的对象:

字符串

整型(或整数)

一个(我猜)整数的数组。

然后,创建这些对象的数组,并将 json 解析为其中。

但是,您的json似乎格式非常糟糕,因为所有成员都应该被命名,例如

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

另一方面,如果 JSON 看起来确实是您描述它的方式,则必须使 Object 数组简单明了,然后在从数据结构中读取它们时强制转换它们。