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

在查看此内容后我尝试了什么(BalusC的答案):

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

但是后来我在日食中遇到了一个错误,说“新的List(){}类型必须实现继承的抽象方法...”,如果我使用快速修复方法,则会得到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并且“结果”不为null。

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

  • 清单类型

    • args = ListOfTypes

      • 列表=空
      • resolveTypes =类型[1]
    • 加载程序= PathClassLoader
    • ownerType0 =空
    • ownerTypeRes = null
    • rawType =类(java.util.ArrayList)
    • rawTypeName =“ java.util.ArrayList”

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

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

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


#1楼

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

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

这样可以避免使用Type对象带来的麻烦,并且如果您确实需要列表,可以始终通过以下方式将数组转换为列表:

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

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

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

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

#2楼

请参阅这篇文章。 Java类型泛型作为GSON的参数

我对此有更好的解决方案。 这是列表的包装器类,因此包装器可以存储列表的确切类型。

public class ListOfJson<T> implements ParameterizedType
{private Class<?> wrapped;public ListOfJson(Class<T> wrapper){this.wrapped = wrapper;}@Overridepublic Type[] getActualTypeArguments(){return new Type[] { wrapped };}@Overridepublic Type getRawType(){return List.class;}@Overridepublic Type getOwnerType(){return null;}
}

然后,代码可以很简单:

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

#3楼

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, "[{...}]");

#4楼

我想补充一种可能性。 如果您不想使用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对象的每个元素。


#5楼

我喜欢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);

#6楼

哭泣,是达到相同结果的另一种方法。 我们使用它的可读性。

而不是写这个难以理解的句子:

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

#7楼

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

#8楼

在我的情况下,@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);

#9楼

请参阅示例2,以了解Gson的“类型”类。

示例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 empTypeListType 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());}
}

#10楼

对于Kotlin而言:

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

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

fun <T> buildType(): Type {return object : TypeToken<ArrayList<T>>() {}.type
}

然后,使用:

val type = buildType<YourMagicObject>()

#11楼

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

#12楼

反序列化通用集合的方法:

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()将编译时类型(在<>之间new TypeToken<...>() {}.getType()捕获到运行时java.lang.reflect.Type对象中。 与只能表示原始(擦除)类型的Class对象不同, Type对象可以表示Java语言中的任何类型,包括泛型的参数化实例化。

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

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

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


#13楼

当它回答我的原始问题时,我已经接受了doc_180的回答,但是如果有人再次遇到此问题,我也将回答问题的第二部分:

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

“ MyClass”类没有“ no args”构造函数,也没有其超类。 在向MyClass及其超类添加简单的“ MyClass()”构造函数后,一切工作正常,包括doc_180建议的List序列化和反序列化。

Google Gson-反序列化列表 class 宾语? (通用类型)相关推荐

  1. Google GSON入门

    在Java世界中,JSON已成为事实上的XML数据交换格式标准,因为它的易用性和传输效率高. 如果您不了解JSON,那就是Javascript对象表示法,这是一种基于文本的数据交换格式,是名称-值的集 ...

  2. 使用google.gson工具时-JSON(谷歌)的使用

    文章目录 1. 创建GSON的两种方法 getAsString()方法返回JsonNull异常问题解决 自定义的返回:setExclusionStrategies 公司项目中使用的实例: 用法详解参考 ...

  3. Android Gson类型转换错误解决 com.google.gson.internal.LinkedTreeMap

    com.google.gson.internal.LinkedTreeMap不能转换到我的类(com.google.gson.internal.LinkedTreeMap cannot be cast ...

  4. Qt/C++ 借助QVariant实现可存储通用类型的容器

    1. 背景 在项目开发过程中,我们可能会遇到这么一种场景:某个或某几个软件组件可以产生许多不同类型的数据,无论是出于性能的考虑,或者是接口简洁性的考虑,这些数据需要被一次性塞到一个类似于数据库的数据容 ...

  5. 问题解决:fastjson 迁移至gson 反序列化错误,看这篇就够!

    0x00 前言 在工作开发总,有一个需求需要将项目中的fastjson迁移至gson,在迁移过程中发现了一个奇葩的bug,如下: org.springframework.http.converter. ...

  6. Google Gson的使用方法及JSON 技术对比

    为什么80%的码农都做不了架构师?>>>    一 .各个JSON技术的简介和优劣 1.json-lib json-lib最开始的也是应用最广泛的json解析工具,json-lib ...

  7. com.google.gson.JsonSyntaxException: java.lang.IllegalStateException: Expected a string but was BEGI

    报错如下: 2020-03-05 18:41:55.161 5576-5576/com.xiayiye.takeout W/System.err: com.google.gson.JsonSyntax ...

  8. 使用Google GSON:额外的赠品:第二部分

    如果您错过了系列的第一篇文章 ,我们将继续上一篇文章的"使用GSON " ,这里是link . 因此,这里我们进行另一系列的讨论. 版本支持 如果要维护一个对象的多个版本以进行JS ...

  9. [TypeScript] 编程实践之1: Google的TypeScript代码风格3:类型

    TypeScript语言规范 3 类型 3.1 Any类型 3.2 基本类型 3.2.1 Number类型 3.2.2 Boolean类型 3.2.3 String类型 3.2.4 Symbol类型 ...

最新文章

  1. python的编程模式-实例解析Python设计模式编程之桥接模式的运用
  2. 内存属于linux文件吗,linux下的/dev/shm是什么? 内存 文件系统
  3. c++远征之模板篇——友元函数、友元类
  4. 让效率“爆表”的49个数据可视化工具
  5. Tell me why -- Declan Galbraith
  6. 用户名 不在 sudoers文件中,此事将被报告。
  7. 【学生党福利】AD 学生免费许可申请
  8. WordPress 图片防盗链的方法
  9. 谱减法matlab程序_从零开始的matlab学习笔记——(2)基本运算
  10. SpringMVC 的运行流程
  11. pandas 索引去重_pandas(一)
  12. android 4.2刷机包,纽曼 K1 Android4.2.1 ROM刷机包 全局提取 精简 流畅 稳定
  13. OpenPLC:符合IEC 61131-3标准的用于网络安全研究的开源工业控制器
  14. 三段式状态机的原理/代码/仿真,1001序列检测实例Verilog
  15. 简谈几种常用的库函数
  16. 服务器空文件夹无法删除怎么办,为什么文件夹删不掉
  17. 深度学习教程(15) | 自然语言处理与词嵌入(吴恩达·完整版)
  18. Redis源码分析(sds)
  19. ERROR: cannot launch node of type [turtlesim /turtlesim_node]: turtlesim
  20. Apache Solr任意文件读取漏洞复现

热门文章

  1. From 《visual C++ 6.0开发工具与调试》
  2. 2021年的“金九银十”你准备好了吗?,腾讯、字节、百度、阿里、快手等一线互联网公司面试真题分享
  3. Skia的SkCamera.cpp的doUpdate() 算法。
  4. 算法------Pow(x, n)
  5. 解决DexOverflowException: Cannot fit requested classes in the main-dex file
  6. 最深刻最通俗的HTTPS 原理详解,图文并茂
  7. java web每天定时执行任务(四步轻松搞定)
  8. 【Gradle】管理库工程release及debug
  9. JS获取当前/指定URL参数
  10. oracle的一知半解