ChatGPT解决这个技术问题 Extra ChatGPT

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

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

在查看 this(BalusC 的回答)后我尝试了什么:

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

但后来我在 Eclipse 中得到一个错误,说“类型 new 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"

args = ListOfTypes list = null resolvedTypes = Type[ 1 ]

列表=空

已解决类型 = 类型 [1]

加载器 = PathClassLoader

所有者类型0 = null

ownerTypeRes = null

rawType = 类 (java.util.ArrayList)

rawTypeName = "java.util.ArrayList"

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

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

编辑:

解决了,看我下面的回答。

您指向的答案使用TokenType。你试过这样吗?
刚刚得到与答案相同的提示。下次我会仔细看看这个例子。 ;)
您可以尝试在类型令牌中实现列表吗?由于您的原始类型是数组列表,您应该尝试数组列表。

D
Daniel Pryden

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

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

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

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

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

有关详细信息,请参阅 documentation for the TypeToken class


在 GSON 的新版本中,TypeToken 构造函数是不公开的,因此在这里你会得到构造函数不可见的错误。在这种情况下你必须做什么?
使用 GSON 的实际版本(2.2.4)它再次工作。您可以在此处访问构造函数。
我的 json 对象以一个对象开头,然后包含我想要的对象数组 { "myObjectArray":[ {....} , {....} , {....} ] },我已经为 {....} 制作了模型文件,如何让这个通用集合代码不假设我的根元素是一个没有制作一个新的嵌套对象文件
需要以下导入 --- import java.lang.reflect.Type;导入 com.google.gson.reflect.TypeToken
如果 YourClass 在代码中是固定的,这很好。如果类在运行时出现怎么办?
E
EpicPandaForce

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

MyClass[] mcArray = gson.fromJson(jsonString, MyClass[].class);

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

List<MyClass> mcList = Arrays.asList(mcArray);

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

要使其成为实际列表(可以修改,请参阅 Arrays.asList() 的限制),只需执行以下操作:

List<MyClass> mcList = new ArrayList<>(Arrays.asList(mcArray));

这很棒!如何将它与反射一起使用?我不知道 MyClass 值,它将被动态定义!
注意:有了这个,请注意 mcList 不是一个完整的列表。很多事情都行不通。
如何将它与泛型一起使用? T[] yourClassList = gson.fromJson(message, T[].class); //不能从类型变量中选择
@MateusViccari 在发表评论时,此答案中的 mcList 只是调用 Arrays.asList 的结果。此方法返回一个列表,其中大多数(如果不是全部)可选方法未实现并引发异常。例如,您不能向该列表添加任何元素。正如后面的编辑所暗示的,Arrays.asList 有局限性,将其包装到实际的 ArrayList 中可以让您获得在许多情况下更有用的列表。
如果您需要在运行时为任意元素类型构造数组类型,您可以使用 Array.newInstance(clazz, 0).getClass(),如 David Wood's answer 中所述。
L
Lii

从 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);
}

示例用法:

String jsonArray = ...
List<User> user = getList(jsonArray, User.class);

TypeToken#getParameterized 看起来比带有匿名子类的 hack 要好得多
我“按原样”复制了您的方法,但它不起作用:编译器说“方法 getParameterized(Class, Class) 对于 TypeToken 类型未定义”。我检查了我的 Gson 版本(2.8.0)和文档,这方面一切都很好......我最终使用了@Happier 解决方案,它工作正常
@leguminator 你导入的 TypeToken 正确吗?你正在使用 java 或 kotlin。我会尝试再次测试
@PhanVanLinh 绝对:我正在使用 Java 并导入了“com.google.gson.reflect.TypeToken”和“java.lang.reflect.Type”。我加倍检查方法实现:它被声明为“public static TypeToken getParameterized(Type rawType, Type... typeArguments)”
这应该是公认的解决方案,简单,它使用 Gson API,并且没有任何黑客攻击。 +1
C
Community

参考这篇文章。 Java Type Generic as Argument for 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;
  }
}

然后,代码可以很简单:

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

mEntity.rulePattern 是什么?
它只是一个用于测试的示例对象。你不需要关心它。使用 toList 方法,一切顺利。
R
Roc Boronat

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

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

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

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

public class YourClassList extends ArrayList<YourClass> {}

并在解析 JSON 时使用它:

List<YourClass> list = new Gson().fromJson(jsonArray, YourClassList.class);

V
Vlad

对于 Kotlin 来说:

import java.lang.reflect.Type
import com.google.gson.reflect.TypeToken
...
val type = object : TypeToken<List<T>>() {}.type

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

fun <T> typeOfList(): Type {
    return object : TypeToken<List<T>>() {}.type
}

然后,使用:

val type = typeOfList<YourMagicObject>()

我已使用您的代码使用具体类型创建此函数:inline fun <reified T> buildType() = object : TypeToken<T>() {}.type!! 并使用 List 类型调用它:buildType<List<YourMagicObject>>()
@coffeemakr 您在这里不需要具体类型。
哦。但是为什么要在 buildType 中创建 ArrayList 的类型标记,并且还要调用具有泛型类型的函数呢?这是一个错字吗? - 这将创建 ArrayList<ArrayList<YourMagicObject>>
@coffeemakr 啊,是的。错字
k
kayz1
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);
}

例子:

getList(MyClass[].class, "[{...}]");

好一个。但这重复了 2 年前写的 DevNG 上面的答案:stackoverflow.com/a/17300003/1339923(并阅读该答案以了解这种方法的注意事项)
j
jellyfish

因为它回答了我最初的问题,所以我已经接受了 doc_180 的回答,但是如果有人再次遇到这个问题,我也会回答我的问题的后半部分:

我描述的 NullPointerError 与 List 本身无关,而是与它的内容有关!

“MyClass”类没有“无参数”构造函数,也没有它的超类。一旦我向 MyClass 及其超类添加了一个简单的“MyClass()”构造函数,一切正常,包括 doc_180 建议的列表序列化和反序列化。


如果你有一个抽象类列表,你会得到同样的错误。我猜这是 GSON 的“无法实例化类”的一般错误消息。
关于添加构造函数的提示帮助我意识到为什么我有所有的空值。我的 JSON 字符串中有“To”和“From”之类的字段名称,但我的对象中的相应字段是小写的“to”和“from”,因此它们被跳过
L
Lii

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

A
Apurva Sharma

我想增加一种可能性。如果您不想使用 TypeToken 并且想将 json 对象数组转换为 ArrayList,那么您可以这样进行:

如果你的 json 结构是这样的:

{

"results": [
    {
        "a": 100,
        "b": "value1",
        "c": true
    },
    {
        "a": 200,
        "b": "value2",
        "c": false
    },
    {
        "a": 300,
        "b": "value3",
        "c": true
    }
]

}

你的班级结构是这样的:

public class ClassName implements Parcelable {

    public ArrayList<InnerClassName> results = new ArrayList<InnerClassName>();
    public static class InnerClassName {
        int a;
        String b;
        boolean c;      
    }
}

然后你可以像这样解析它:

Gson gson = new Gson();
final ClassName className = gson.fromJson(data, ClassName.class);
int currentTotal = className.results.size();

现在您可以访问 className 对象的每个元素。


S
Shashanth

有关 Gson 的“类型”类理解,请参阅示例 2。

示例 1:在这个 deserilizeResturant 中,我们使用了 Employee[] 数组并获取了详细信息

public static void deserializeResturant(){

       String empList ="[{\"name\":\"Ram\",\"empId\":1},{\"name\":\"Surya\",\"empId\":2},{\"name\":\"Prasants\",\"empId\":3}]";
       Gson gson = new Gson();
       Employee[] emp = gson.fromJson(empList, Employee[].class);
       int numberOfElementInJson = emp.length();
       System.out.println("Total JSON Elements" + numberOfElementInJson);
       for(Employee e: emp){
           System.out.println(e.getName());
           System.out.println(e.getEmpId());
       }
   }

示例 2:

//Above deserilizeResturant used Employee[] array but what if we need to use List<Employee>
public static void deserializeResturantUsingList(){

    String empList ="[{\"name\":\"Ram\",\"empId\":1},{\"name\":\"Surya\",\"empId\":2},{\"name\":\"Prasants\",\"empId\":3}]";
    Gson gson = new Gson();

    // Additionally we need to se the Type then only it accepts List<Employee> which we sent here empTypeList
    Type empTypeList = new TypeToken<ArrayList<Employee>>(){}.getType();


    List<Employee> emp = gson.fromJson(empList, empTypeList);
    int numberOfElementInJson = emp.size();
    System.out.println("Total JSON Elements" + numberOfElementInJson);
    for(Employee e: emp){
        System.out.println(e.getName());
        System.out.println(e.getEmpId());
    }
}

P
Peterstev Uremgba

使用 Kotlin,您可以获得所有自定义 Serializable 类型的通用 MutableList 类型

private fun <T : Serializable> getGenericList(
    sharedPreferences: SharedPreferences,
    key: String,
    clazz: KClass<T>
): List<T> {
    return sharedPreferences.let { prefs ->
        val data = prefs.getString(key, null)
        val type: Type = TypeToken.getParameterized(MutableList::class.java, clazz.java).type
        gson.fromJson(data, type) as MutableList<T>
    }
}

你可以调用这个函数

getGenericList.(sharedPrefObj, sharedpref_key, GenericClass::class)

A
Andrei

在我的情况下,@uncaught_exceptions 的答案不起作用,我不得不使用 List.class 而不是 java.lang.reflect.Type

String jsonDuplicatedItems = request.getSession().getAttribute("jsonDuplicatedItems").toString();
List<Map.Entry<Product, Integer>> entries = gson.fromJson(jsonDuplicatedItems, List.class);

o
oleg.cherednik

我为此案例创建了 GsonUtils 库。我将它添加到 Maven 中央存储库中。

Map<String, SimpleStructure> expected = new HashMap<>();
expected.put("foo", new SimpleStructure("peperoni"));

String json = GsonUtils.writeValue(expected);

Map<String, SimpleStructure> actual = GsonUtils.readMap(json, String.class, SimpleStructure.class);

s
spenibus

我喜欢 kays1 的答案,但我无法实现它。所以我使用他的概念构建了我自己的版本。

public class JsonListHelper{
    public static final <T> List<T> getList(String json) throws Exception {
        Gson gson = new GsonBuilder().setDateFormat("yyyy-MM-dd HH:mm:ss").create();
        Type typeOfList = new TypeToken<List<T>>(){}.getType();
        return gson.fromJson(json, typeOfList);
    }
}

用法:

List<MyClass> MyList= JsonListHelper.getList(jsonArrayString);

这肯定行不通,因为您试图在编译时使用 T 。这将有效地反序列化为 StringMap 列表,不是吗?