我想通过 Google Gson 传输列表对象,但我不知道如何反序列化泛型类型。
看了 这个 之后我尝试了什么(BalusC 的回答):
MyClass mc = new Gson().fromJson(result, new List<MyClass>() {}.getClass());
但后来我在 Eclipse 中收到一条错误消息,说“新类型 List<MyClass>() {}
必须实现继承的抽象方法……”,如果我使用快速修复,我会得到超过 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
和 result
不为空。
我用调试器检查了 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 期间应该发生的运行时异常。我做错了(上面未显示),就像示例中一样,但根本没有得到该异常。所以我按照用户指南的建议更改了序列化。没有帮助,虽然。
编辑:
已解决,请参阅下面的答案。
原文由 jellyfish 发布,翻译遵循 CC BY-SA 4.0 许可协议
反序列化泛型集合的方法:
由于评论中有几个人提到了它,这里解释一下如何使用
TypeToken
类。 The constructionnew TypeToken<...>() {}.getType()
captures a compile-time type (between the<
and>
) into a runtimejava.lang.reflect.Type
object.与Class
对象不同,它只能表示原始(擦除)类型,Type
对象可以表示 Java 语言中的任何类型,包括泛型类型的参数化实例化。TypeToken
类本身没有公共构造函数,因为您不应该直接构造它。相反,您总是构造一个匿名子类(因此{}
,这是该表达式的必要部分)。由于类型擦除,
TypeToken
类只能捕获在编译时完全已知的类型。 (也就是说,您不能对类型参数执行new TypeToken<List<T>>() {}.getType()
T
。)有关详细信息,请参阅
TypeToken
类的文档。