Gson中的TypeToken与泛型擦除详情

目录
  • 问题
  • TypeToken是什么
  • 其它使用场景

问题

在Java的json框架中,Gson是使用得比较广泛的一个,其Gson类提供了toJson()fromJson()方法,分别用来序列化与反序列化。

json序列化用得最多的场景是在调用外部服务接口时,大致如下:

@Data
@AllArgsConstructor
public class Response<T>{
    int code;
    String message;
    T body;
}

@Data
@AllArgsConstructor
public class PersonInfo{
    long id;
    String name;
    int age;
}

/**
 * 服务端
 */
public class Server {
    public static String getPersonById(Long id){
        PersonInfo personInfo = new PersonInfo(1234L, "zhangesan", 18);
        Response<PersonInfo> response = new Response<>(200, "success", personInfo);
        //序列化
        return new Gson().toJson(response);
    }
}

/**
 * 客户端
 */
public class Client {
    public static void getPerson(){
        String responseStr = Server.getPersonById(1234L);
        //反序列化
        Response<PersonInfo> response = new Gson().fromJson(responseStr, new TypeToken<Response<PersonInfo>>(){}.getType());
        System.out.println(response);
    }
}

由于大多数接口设计中,都会有统一的响应码结构,因此大多项目都会像上面一样,设计一个通用Response类来对应这种统一响应码结构,是很常见的情况。

但会发现,在反序列化过程中,传入目标类型时,使用了一段很奇怪的代码,即new TypeToken<Response<PersonInfo>>(){}.getType(),那它是什么?为啥要使用它?

TypeToken是什么

为什么要使用TypeToken呢?我们直接使用Response<PersonInfo>.class行不行?如下:

可以发现,java并不允许这么使用!

那传Response.class呢?如下:

可以发现,代码能跑起来,但是Body变成了LinkedHashMap类型,这是因为传给gson的类型是Response.class,gson并不知道body属性是什么类型,那它只能使用LinkedHashMap这个默认的json对象类型了。

这就是TypeToken由来的原因,对于带泛型的类,使用TypeToken才能得到准确的类型信息,那TypeToken是怎么取到准确的类型的呢?

首先,new TypeToken<Response<PersonInfo>>(){}.getType()实际上是定义了一个匿名内部类的对象,然后调用了这个对象的getType()方法。

看看getType()的实现,如下:

逻辑也比较简单,先通过getGenericSuperclass()获取了此对象的父类,即TypeToken<Response<PersonInfo>>,然后又通过getActualTypeArguments()[0]获取了实际类型参数,即Response<PersonInfo>

额,逻辑看起来说得通,但不是说Java泛型会擦除吗?这里不会擦除?

从所周知,java泛型擦除发生在编译期,ok,那我模拟上面的原理,写个空类继承TypeToken<Response<PersonInfo>>,然后编译这个类之后再反编译一下,看类型到底擦除没!

public class PersonResponseTypeToken extends TypeToken<Response<PersonInfo>> {

}

反编译结果如下:

也就是说,被继承的父类上的泛型是不擦除的。

其它使用场景

有时为了编程的方便,经常会有框架将远程调用接口化,类似下面这样:

public class RemoteUtil {
    private static final ConcurrentMap<Class, Object> REMOTE_CACHE = new ConcurrentHashMap<>();

    public static <T> T get(Class<T> clazz) {
        return clazz.cast(REMOTE_CACHE.computeIfAbsent(clazz, RemoteUtil::getProxyInstance));
    }

    private static Object getProxyInstance(Class clazz) {
        return Proxy.newProxyInstance(ClassLoader.getSystemClassLoader(), new Class[]{clazz}, (proxy, method, args) -> {
            Gson gson = new Gson();
            String path = method.getAnnotation(RequestMapping.class).path()[0];
            HttpURLConnection conn = null;
            try {
                conn = (HttpURLConnection) new URL("http://localhost:8080/" + path).openConnection();
                conn.setRequestMethod("POST");
                conn.setDoOutput(true);
                conn.setDoInput(true);
                conn.connect();
                //设置请求数据
                JsonObject requestBody = new JsonObject();
                try (Writer out = new OutputStreamWriter(conn.getOutputStream(), StandardCharsets.UTF_8)) {
                    int i = 0;
                    for (Parameter parameter : method.getParameters()) {
                        String name = parameter.getAnnotation(RequestParam.class).name();
                        requestBody.add(name, gson.toJsonTree(args[i]));
                        i++;
                    }
                    out.write(requestBody.toString());
                }
                //获取响应数据
                if (conn.getResponseCode() != HttpURLConnection.HTTP_OK) {
                    throw new RuntimeException("远程调用发生异常:url:" + conn.getURL() + ", requestBody:" + requestBody);
                }
                String responseStr = IOUtils.toString(conn.getInputStream(), StandardCharsets.UTF_8);
                //响应结果反序列化为具体对象
                return gson.fromJson(responseStr, method.getReturnType());
            } finally {
                if (conn != null) {
                    conn.disconnect();
                }
            }
        });
    }

}

public interface PersonApi {
    @RequestMapping(path = "/person")
    Response<PersonInfo> getPersonById(@RequestParam(name = "id") Long id);
}

public class Client {

    public static void getPerson() {
        Response<PersonInfo> response = RemoteUtil.get(PersonApi.class).getPersonById(1234L);
        System.out.println(response.getBody());
    }
}

这样做的好处是,开发人员不必再关心如何发远程请求了,只需要定义与调用接口即可。

但上面调用过程中会有一个问题,就是获取的response对象中body属性是LinkedHashMap,原因是gson反序列化时是通过method.getReturnType()来获取返回类型的,而返回类型中的泛型会被擦除掉。

要解决这个问题也很简单,和上面TypeToken一样的道理,定义一个空类PersonResponse来继承Response<PersonInfo>,然后将返回类型定义为PersonResponse

如下:

public class PersonResponse extends Response<PersonInfo> {
}

public interface PersonApi {
    @RequestMapping(path = "/person")
    PersonResponse getPersonById(@RequestParam(name = "id") Long id);
}

然后你就会发现,gson可以正确识别到body属性的类型了。

到此这篇关于Gson中的TypeToken与泛型擦除详情的文章就介绍到这了,更多相关Gson TypeToken内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Android Gson基本用法学习

    目录 1. 导入Android Studio工程 2. 简单的 Java Object 序列化/反序列化 序列化 反序列化 3. 嵌套 Java Object 的序列化/反序列化 4. Array 和 List 的序列化/反序列化 序列化 反序列化 1 Array的反序列化 2 List的反序列化 5. Map 和 Set 的序列化/反序列化 7. 控制序列化/反序列化 的变量名称 8. 序列化/反序列化过程中忽略某些变量 Gson是谷歌官方推出的支持 JSON -- Java Object 相

  • Gson序列化指定忽略字段的三种写法详解

    目录 1. transient关键字 2. expose注解 3. 自定义排查策略ExclusionStrategy 在我们日常使用json序列化框架过程中,经常会遇到在输出json字符串时,忽略某些字段,那么在Gson框架中,要想实现这种方式,可以怎么处理呢? 本文介绍几种常见的姿势 1. transient关键字 最容易想到的case,就是直接借助jdk的transient关键字来修饰不希望输出的对象,如 @Data @AllArgsConstructor @NoArgsConstructo

  • 解决Android Studio4.1没有Gsonfomat插件,Plugin “GsonFormat” is incompatible的问题

    前言 前几天 在自己的 笔记本上把android studio 升级到4.1了 一直没有使用Gsonfomat插件所以没有发现问题! 今天使用GsonFormat,发现GsonFormat没有了,在android studio 的插件里搜不到于是百度搜官网去下载 GsonFormat插件官网地址 下载jar包以后,本地安装插件,还是不行,直接报错了! 报错Plugin "GsonFormat" is incompatible (supported only in IntelliJ ID

  • 使用Gson将字符串转换成JsonObject和JsonArray

    目录 Gson将字符串转JsonObject和JsonArray 以下均利用Gson来处理 JSONObject与JSON互转 引入 jar , 此处引入 com.alibaba.fastjson 版本的jar包 建立测试类对象 转换 Gson将字符串转JsonObject和JsonArray 以下均利用Gson来处理 1.将bean转换成Json字符串: public static String beanToJSONString(Object bean) {         return ne

  • IDEA使用GsonFormat完成JSON和JavaBean之间的转换

    最近一直在对接接口,上游返回的都是 JSON 数据,我们需要将这些数据进行保存,我们可以解析成 Map 通过 key 的方式进行获取,然后 set 到实体类对象中,说到这里我开始想吐了,这样就造成了代码过多,没有可读性,如果有100个值,要 get 100次, set 100次吗? 所以最简单的方式是封装成对象,通过对象操作工具进行对象中属性值的映射,但是封装对象过程又繁琐了,属性过多极大的浪费时间,记得初中历史学过的一段话,人和动物最根本的区别就是会不会制造和使用工具,大神和菜鸟之间的差距也莫

  • Gson之toJson和fromJson方法的具体使用

    目录 1.toJson()方法是实现从java实体到Json相关对象的方法 2.fromJson()方法来实现从Json相关对象到java实体的方法 Gson是Google的一个开源项目,可以将Java对象转换成JSON,也可能将JSON转换成Java对象. Gson里最重要的对象有2个Gson 和 GsonBuilder Gson有2个最基本的方法 toJson() – 转换java 对象到JSON fromJson() – 转换JSON到java对象 引入依赖:pom.xml文件中加入 <!

  • JSON中fastjson、jackson、gson如何选择

    目录 1 fastjson 2 jsoncode 3 jsonpath JSON具有表达简洁.层级清晰的特点,目前广泛应用在数据的通信传输中,尤其前后端的交互,几乎都是使用JSON实现的.例如下面的数据: { "code" : 0, "kind" : "Electronics", "list" : [{ "name" : "computer", "price" : 4

  • Gson中的TypeToken与泛型擦除详情

    目录 问题 TypeToken是什么 其它使用场景 问题 在Java的json框架中,Gson是使用得比较广泛的一个,其Gson类提供了toJson()与fromJson()方法,分别用来序列化与反序列化. json序列化用得最多的场景是在调用外部服务接口时,大致如下: @Data @AllArgsConstructor public class Response<T>{ int code; String message; T body; } @Data @AllArgsConstructor

  • Java中的纸老虎之泛型

    目录 一. 泛型的定义 二. 为什么要用到泛型 三. 泛型的写法 四. 泛型的使用实例 1. 求最大值 2. 优化 五. 通配符 1. 基本写法 2. 上界 3. 下界 六. 泛型的限制 泛型,其实算是Java当中比较难的语法了,很多人一开始都对其一知半解,也很害怕阅读带泛型的源码,虽然看起来语法很难,但当你理解后会觉得很简单,其实只是一个纸老虎罢了.下面,我将会用非常简单易懂的方式带你去理解它,相信你在认真看完后会有非常大的收获,从此不会再畏惧它! 一. 泛型的定义 这里大家可以不必去看网上的

  • Java 泛型考古 泛型擦除 包装类详细解析

    目录 一. 什么是泛型 二. 为什么要有泛型 ? 示例 三.泛型考古 四.泛型擦除 五.包装类 六.装箱拆箱 一. 什么是泛型 泛型(generic type)其本质是将类型参数化,也就是说所操作的数据类型被指定为一个参数这种参数类型可以用在类.接口和方法的创建中,分别称为泛型类.泛型接口.泛型方法. 二. 为什么要有泛型 ? 之前写过MyArrayList顺序表,这个类当时自己在实现的时候只能用一种类型来表示,也就是用的时候自己实现的MyArrayList只能应用于一种类型,要想应用于其他类型

  • Swift中风味各异的类型擦除实例详解

    目录 前言 什么时候需要类型擦除? 通用包装器类型擦除 闭包类型擦除 结语 前言 Swift的总体目标是既强大到可以用于底层系统编程,又足够容易让初学者学习,这有时会导致相当有趣的情况——当Swift的类型系统的力量要求我们部署相当高级的技术来解决乍一看可能更微不足道的问题. 大多数Swift开发人员会在某一时刻或另一时刻(通常是马上,而不是日后)会遇到这样一种情况,即需要某种形式的类型擦除才能引用通用协议.从本周开始,让我们看一下是什么使类型擦除在Swift中成为必不可少的技术,然后继续探索实

  • 关于泛型擦除问题的解决--Mybatis查询类型转换

    目录 概念介绍 问题案例 原因分析 解决方案 总结 概念介绍 Java语言的泛型采用的是擦除法实现的伪泛型,泛型信息(类型变量.参数化类型)编译之后通通被除掉了.使用擦除法的好处就是实现简单.非常容易Backport,运行期也能够节省一些类型所占的内存空间. 而擦除法的坏处就是,通过这种机制实现的泛型远不如真泛型灵活和强大.Java选取这种方法是一种折中,因为Java最开始的版本是不支持泛型的,为了兼容以前的库而不得不使用擦除法. 验证擦除,我们编写下面代码: public class Eras

  • Gson中@JsonAdater注解的几种方式总结

    目录 Gson @JsonAdater注解的几种方式 总结 问题描述 方式一 方式二-write原样 方式三-简单写法 Gson注解 @SerializedName Expose Gson @JsonAdater注解的几种方式 总结 可以通过自定义TypeAdapter和TypeAdapterFactory的方式,自定义gson的序列化和反序列规则,TypeAdapterFactory可以拿到上下文gson.TokenType类型: 也可以通过继承JsonReader重新写一次代码,在begin

  • Java编程探索之泛型擦除实例解析

    1.问题引出 源码: public static void main(String[] args) { List<Integer> a = new ArrayList<Integer>(); List<String> b = new ArrayList<String>(); System.out.println(a.getClass() == b.getClass());//结果true } 编译后L public static void main(Stri

  • 详解Java中的 枚举与泛型

    详解Java中的 枚举与泛型 一:首先从枚举开始说起 枚举类型是JDK5.0的新特征.Sun引进了一个全新的关键字enum来定义一个枚举类.下面就是一个典型枚举类型的定义: public enum Color{ RED,BLUE,BLACK,YELLOW,GREEN } 显然,enum很像特殊的class,实际上enum声明定义的类型就是一个类. 而这些类都是类库中Enum类的子类(Java.lang.Enum).它们继承了这个Enum中的许多有用的方法.我们对代码编译之后发现,编译器将 enu

  • Java中方法名称和泛型相同的用法示例

    本文实例讲述了Java中方法名称和泛型相同的用法.分享给大家供大家参考,具体如下: 一 点睛 Java中,方法的名称可以用泛型替代. 二 实战 1 代码 public class SupGent { public class A<E> { E t; public A( E t ) { this.t = t; } public E E() { //采用了泛型E,碰巧方法名称也是E,只不过不要弄混淆,有点像宏替换 return t; } } public class B<E> exte

  • Spring 中优雅的获取泛型信息的方法

    简介 Spring 源码是个大宝库,我们能遇到的大部分工具在源码里都能找到,所以笔者开源的 mica 完全基于 Spring 进行基础增强,不重复造轮子.今天我要分享的是在 Spring 中优雅的获取泛型. 获取泛型 自己解析 我们之前的处理方式,代码来源 vjtools(江南白衣). /** * 通过反射, 获得Class定义中声明的父类的泛型参数的类型. * * 注意泛型必须定义在父类处. 这是唯一可以通过反射从泛型获得Class实例的地方. * * 如无法找到, 返回Object.clas

随机推荐