Java-TypeToken原理及泛型擦除(1)

本文解释了TypeToken在Gson反序列化中的作用,解决因运行时泛型擦除导致的类型获取问题。通过匿名类和TypeToken的子类来保存泛型信息,以及详细分析了Gson如何通过TypeImpl获取实际类型参数。
摘要由CSDN通过智能技术生成

同时正正有这个这么“坑”的机制,令到我们无法在运行期间随心所欲的获取到泛型参数的具体类型。

TypeToken


使用

使用过Gson的同学都知道在反序列化时需要定义一个TypeToken类型,像这样

private Type type = new TypeToken<List<Map<String, Foo>>>(){}.getType();

//调用fromJson方法时把type传过去,如果type的类型和json保持一致,则可以反序列化出来

gson.fromJson(json, type);

三个问题

为什么要用TypeToken来定义反序列化的类型?正如上面说的,如果直接把List<Map<String, Foo>>的类型传过去,但是因为运行时泛型被擦除了,所以得到的其实是List,那么后面的Gson就不知道要转成Map<String, Foo>类型了,这时Gson会默认转成LinkedTreeMap类型。

为什么带有大括号{}?这个大括号就是精髓所在。大家都知道,在Java语法中,在这个语境,{}是用来定义匿名类,这个匿名类是继承了TypeToken类,它是TypeToken的子类。

为什么要通过子类来获取泛型的类型?这是TypeToken能够获取到泛型类型的关键,这是一个巧妙的方法。这个想法是这样子的,既然像List这样中的泛型会被擦除掉,那么我用一个子类SubList extends List这样的话,在JVM内部中会不会把父类泛型的类型给保存下来呢?我这个子类需要继承的父类的泛型都是已经确定了的呀,果然,JVM是有保存这部分信息的,它是保存在子类的Class信息中,具体看:https://stackoverflow.com/questions/937933/where-are-generic-types-stored-in-java-class-files 那么我们怎么获取这部分信息呢?还好,Java有提供API出来:

Type mySuperClass = foo.getClass().getGenericSuperclass();

Type type = ((ParameterizedType)mySuperClass).getActualTypeArguments()[0];

System.out.println(type);

分析一下这段代码,Class类的getGenericSuperClass()方法的注释是:

Returns the Type representing the direct superclass of the entity (class, interface, primitive type or void) represented by thisClass. If the superclass is a parameterized type, the Type object returned must accurately reflect the actual type parameters used in the source code. The parameterized type representing the superclass is created if it had not been created before. See the declaration of ParameterizedType for the semantics of the creation process for parameterized types. If thisClass represents either theObject class, an interface, a primitive type, or void, then null is returned. If this object represents an array class then theClass object representing theObject class is returned

概括来说就是对于带有泛型的class,返回一个ParameterizedType对象,对于Object、接口和原始类型返回null,对于数 组class则是返回Object.class。ParameterizedType是表示带有泛型参数的类型的Java类型,JDK1.5引入了泛型之 后,Java中所有的Class都实现了Type接口,ParameterizedType则是继承了Type接口,所有包含泛型的Class类都会实现 这个接口。

自己调试一下就知道它返回的是什么了。

原理

核心的方法就是刚刚说的那两句,剩下的就很简单了。我们看看TypeToken的getType方法

public final Type getType() {

//直接返回type

return type;

}

看type的初始化

//注意这里用了protected关键字,限制了只有子类才能访问

protected TypeToken() {

this.type = getSuperclassTypeParameter(getClass());

this.rawType = (Class<? super T>) G s o n Gson GsonTypes.getRawType(type);

this.hashCode = type.hashCode();

}

//getSuperclassTypeParameter方法

//这几句就是上面的说到

static Type getSuperclassTypeParameter(Class<?> subclass) {

Type superclass = subclass.getGenericSuperclass();

if (superclass instanceof Class) {

throw new RuntimeException(“Missing type parameter.”);

}

ParameterizedType parameterized = (ParameterizedType) superclass;

//这里注意一下,返回的是Gson自定义的,在 G s o n Gson GsonTypes里面定义的TypeImpl等,这个类都是继承Type的。

return G s o n Gson GsonTypes.canonicalize(parameterized.getActualTypeArguments()[0]);

}

最后

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Java工程师,想要提升技能,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助。

因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门!

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

  • 10
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值