Google Gson - 反序列化列表<class>对象?(通用型)

jel*_*ish 409 java generics json gson

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

我看后试了一下这个(BalusC的答案):

MyClass mc = new Gson().fromJson(result, new List<MyClass>(){}.getClass());
Run Code Online (Sandbox Code Playgroud)

但是我在eclipse中得到一个错误,说"类型新List(){}必须实现继承的抽象方法......"如果我使用快速修复,我会得到一个超过20个方法存根的怪物.

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

编辑:

我现在有

Type listType = new TypeToken<List<MyClass>>()
                {
                }.getType();

MyClass mc = new Gson().fromJson(result, listType);
Run Code Online (Sandbox Code Playgroud)

但是,我在"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)
Run Code Online (Sandbox Code Playgroud)

抓JsonParseExceptions和"结果"是不为空.

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

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

所以似乎"getClass"调用无法正常工作.有什么建议...?

编辑2:我查看了Gson用户指南.它提到了在将泛型类型解析为Json时应该发生的运行时异常.我做的是"错误的"(上面未显示),就像在示例中一样,但根本没有得到那个例外.所以我按照建议的用户指南更改了序列化.但是,没有帮助.

编辑3:解决了,请参阅下面的答案.

unc*_*ons 905

反序列化泛型集合的方法:

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);
Run Code Online (Sandbox Code Playgroud)

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

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

由于类型擦除,TypeToken该类只能捕获在编译时完全已知的类型.(也就是说,你不能new TypeToken<List<T>>() {}.getType()为类型参数做T.)

有关更多信息,请参阅该类文档TypeToken.

  • 在新版本的GSON中,TypeToken构造函数不是公共的,因此在这里你得到构造函数不可见的错误.在这种情况下你需要做什么? (30认同)
  • 使用GSON(2.2.4)的实际版本,它再次起作用.您可以在此处访问构造函数. (7认同)
  • 按照Imports要求--- import java.lang.reflect.Type; 导入com.google.gson.reflect.TypeToken (6认同)
  • 如果在代码中修复了YourClass,这很好.如果课程在运行时出现怎么办? (4认同)
  • 类型是建议许多进口,我应该导入什么? (2认同)

Dev*_*vNG 259

另一种方法是使用数组作为类型,例如:

MyClass[] mcArray = gson.fromJson(jsonString, MyClass[].class);
Run Code Online (Sandbox Code Playgroud)

这样你可以避免使用Type对象的所有麻烦,如果你真的需要一个列表,你总是可以通过以下方式将数组转换为列表:

List<MyClass> mcList = Arrays.asList(mcArray);
Run Code Online (Sandbox Code Playgroud)

恕我直言,这更具可读性.

并使其成为一个实际列表(可以修改,请参阅限制Arrays.asList()),然后执行以下操作:

List<MyClass> mcList = new ArrayList<>(Arrays.asList(mcArray));
Run Code Online (Sandbox Code Playgroud)

  • 这很棒!我如何用反射来使用它?我不知道`MyClass`值,它将动态定义! (3认同)
  • 如何使用泛型?`T [] yourClassList = gson.fromJson(message,T [] .class);`//无法从类型变量中选择 (3认同)
  • 注意:有了这个,请注意 mcList 不是一个完整的列表。很多事情都行不通。 (2认同)
  • @MateusViccari在评论时,这个答案中的`mcList`只是调用`Arrays.asList`的结果.此方法返回一个列表,大多数(如果不是所有)可选方法都未实现,并抛出异常.例如,您无法向该列表添加任何元素.正如后面的编辑所暗示的那样,`Arrays.asList`有局限性,并将其包装到一个实际的`ArrayList`中,这样你就可以得到一个在很多情况下更有用的列表. (2认同)
  • 如果您需要在运行时为任意元素类型构造数组类型,您可以使用 `Array.newInstance(clazz, 0).getClass()`,如 [David Wood 的回答](https://stackoverflow.com/) 中所述问题/5554217/google-gson-deserialize-listclass-object-generic-type/36338377#36338377)。 (2认同)

Hap*_*ier 27

请参阅这篇文章. Java Type Generic作为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;
  }
}
Run Code Online (Sandbox Code Playgroud)

然后,代码可以很简单:

public static <T> List<T> toList(String json, Class<T> typeClass)
{
    return sGson.fromJson(json, new ListOfJson<T>(typeClass));
}
Run Code Online (Sandbox Code Playgroud)


Roc*_*nat 23

Wep,另一种实现相同结果的方法.我们使用它是为了它的可读性.

而不是做这个难以阅读的句子:

Type listType = new TypeToken<ArrayList<YourClass>>(){}.getType();
List<YourClass> list = new Gson().fromJson(jsonArray, listType);
Run Code Online (Sandbox Code Playgroud)

创建一个扩展对象List的空类:

public class YourClassList extends ArrayList<YourClass> {}
Run Code Online (Sandbox Code Playgroud)

并在解析JSON时使用它:

List<YourClass> list = new Gson().fromJson(jsonArray, YourClassList.class);
Run Code Online (Sandbox Code Playgroud)


Pha*_*inh 22

既然如此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);
}
Run Code Online (Sandbox Code Playgroud)

使用示例

String jsonArray = ...
List<User> user = getList(jsonArray, User.class);
Run Code Online (Sandbox Code Playgroud)

  • `TypeToken#getParameterized` 看起来比使用匿名子类的黑客更好 (2认同)
  • 就其价值而言,使用匿名“TypeToken”子类的解决方案并不是真正的黑客,而是 Gson 官方推荐的用法(请参阅其 Javadoc 和 [用户指南](https://github.com/google/gson/blob /master/UserGuide.md#TOC-序列化和反序列化-通用类型))。主要优点是它确保您构造正确的参数化类型。`TypeToken.getParameterized` 无法在编译时执行任何验证,因此您可以构造诸如 `List&lt;String, String, String&gt;` 或 `Map&lt;Integer&gt;` 之类的类型;最新的 Gson 版本至少在运行时检测到这些不良类型。 (2认同)

kay*_*yz1 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);
}
Run Code Online (Sandbox Code Playgroud)

例:

getList(MyClass[].class, "[{...}]");
Run Code Online (Sandbox Code Playgroud)


Cha*_*ham 7

对于Kotlin而言:

import java.lang.reflect.Type
import com.google.gson.reflect.TypeToken
...
val type = object : TypeToken<ArrayList<T>>() {}.type
Run Code Online (Sandbox Code Playgroud)

或者,这是一个有用的功能:

fun <T> buildType(): Type {
    return object : TypeToken<ArrayList<T>>() {}.type
}
Run Code Online (Sandbox Code Playgroud)

然后,使用:

val type = buildType<YourMagicObject>()
Run Code Online (Sandbox Code Playgroud)


jel*_*ish 6

当它回答我原来的问题时,我接受了doc_180的回答,但是如果有人再次遇到这个问题,我也会回答我问题的第二部分:

我描述的NullPointerError与List本身无关,但与其内容有关!

"MyClass"类没有"no args"构造函数,也没有超类一.一旦我向MyClass及其超类添加了一个简单的"MyClass()"构造函数,一切都运行正常,包括doc_180建议的List序列化和反序列化.


Dav*_*ood 6

这是一个适用于动态定义类型的解决方案。诀窍是使用 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; 
}
Run Code Online (Sandbox Code Playgroud)