使用 Gson 反序列化 List<T> 对象?

2022-08-31 04:22:05

我想通过Google Gson传输列表对象,但我不知道如何反序列化泛型类型。

看了这个(BalusC的答案)后我尝试了什么:

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

但后来我在Eclipse中收到一个错误,说“新类型必须实现继承的抽象方法......”如果我使用快速修复,我会得到一个超过20个方法存根的怪物。List<MyClass>() {}

我很确定有一个更简单的解决方案,但我似乎找不到它!

现在我有这个:

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)

我确实捕获并且不为空。JsonParseExceptionsresult

我检查了调试器,得到了以下内容:listType

  • list Type
    • args = ListOfTypes
      • list = null
      • resolvedTypes = Type[ 1 ]
    • loader = PathClassLoader
    • ownerType0 = null
    • ownerTypeRes = null
    • rawType = Class (java.util.ArrayList)
    • rawTypeName = "java.util.ArrayList"

因此,调用似乎无法正常工作。有什么建议吗?getClass

我已经检查了Gson用户指南。它提到了在将泛型类型解析为 Json 期间应发生的运行时异常。我做错了(上面没有显示),就像在示例中一样,但根本没有得到这个例外。因此,我按照用户指南中的建议更改了序列化。不过没有帮助。

编辑:

解决了,请参阅下面的答案。


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

由于评论中的几个人提到了它,因此以下是如何使用该类的解释。该构造将编译时类型(介于 和 之间)捕获到运行时对象中。与只能表示原始(擦除)类型的对象不同,该对象可以表示 Java 语言中的任何类型,包括泛型类型的参数化实例化。TypeTokennew TypeToken<...>() {}.getType()<>java.lang.reflect.TypeClassType

类本身没有公共构造函数,因为您不应该直接构造它。相反,您总是构造一个匿名子类(因此 ,这是此表达式的必要部分)。TypeToken{}

由于类型擦除,该类只能捕获在编译时完全已知的类型。(也就是说,您不能对类型参数执行 操作。TypeTokennew TypeToken<List<T>>() {}.getType()T

有关更多信息,请参见 TypeToken 类的文档


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