以前就了解过Java泛型的实现是不完整的,最近在做一些代码重构的时候遇到一些Java泛型类型擦除的问题,简单的来说,Java泛型中所指定的类型在编译时会将其去除,因此List 和 List 在编译成字节码的时候实际上是一样的。因此java泛型只能做到编译期检查的功能,运行期间就不能保证类型安全。我最近遇到的一个问题如下:

假设有两个bean类

  1. /** Test. */
  2. @Data
  3. @NoArgsConstructor
  4. @AllArgsConstructor
  5. public static class Foo {
  6. public String name;
  7. }
  8. /** Test. */
  9. @Data
  10. @NoArgsConstructor
  11. @AllArgsConstructor
  12. public static class Dummy {
  13. public String name;
  14. }

以及另一个对象

  1. @NoArgsConstructor
  2. @AllArgsConstructor
  3. @Data
  4. public static class Spec<T> {
  5. public String spec;
  6. public T deserializeTo() throws JsonProcessingException {
  7. var mapper = new ObjectMapper();
  8. return (T) mapper.readValue(spec, Foo.class);
  9. }
  10. }

可以看到Spec对象中保存了以上两种类型json序列化后的字符串,并提供了方法将string spec 反序列化成相应的类型,比较理想的方式是在反序列化的方法中能够获取到参数类型 T 的实际类型,理论上运行时Spec类型是确定了,因此T也应该是确定的,但是因为类型擦除,所以实际上获取不到他的类型。

按照以下尝试 通过((ParameterizedType) getClass().getGenericSuperclass()).getActualTypeArguments()获取泛型类型,经过测试是获取不到的

  1. @Test
  2. public void test() throws JsonProcessingException {
  3. var foo = new Foo("foo");
  4. var spec = new Spec<Foo>(mapper.writeValueAsString(foo));
  5. var deserialized = spec.deserializeTo();
  6. Assertions.assertTrue(deserialized instanceof Foo);
  7. }
  8. @NoArgsConstructor
  9. @AllArgsConstructor
  10. @Data
  11. public static class Spec<T> {
  12. public String spec;
  13. private Class<T> getSpecClass() {
  14. return (Class<T>)
  15. ((ParameterizedType) getClass().getGenericSuperclass())
  16. .getActualTypeArguments()[0];
  17. }
  18. public T deserializeTo() throws JsonProcessingException {
  19. var mapper = new ObjectMapper();
  20. System.out.println(spec);
  21. return (T) mapper.readValue(spec, getSpecClass());
  22. }
  23. }

会有以下的错误

java.lang.ClassCastException: class java.lang.Class cannot be cast to class java.lang.reflect.ParameterizedType (java.lang.Class and java.lang.reflect.ParameterizedType are in module java.base of loader 'bootstrap')

有两种办法来绕过这个问题

第一种比较简单,就是在创建spec对象时,直接把类型的class传进来,这样就可以直接使用。

第二种是创建spec的子类中使用这个方法就可以获取泛型的类型

  1. @Data
  2. public abstract static class AbstractSpec<T> {
  3. public String spec;
  4. public AbstractSpec(String spec) {
  5. this.spec = spec;
  6. }
  7. private Class<T> getSpecClass() {
  8. return (Class<T>)
  9. ((ParameterizedType) getClass().getGenericSuperclass())
  10. .getActualTypeArguments()[0];
  11. }
  12. public T deserializeTo() throws JsonProcessingException {
  13. var mapper = new ObjectMapper();
  14. System.out.println(spec);
  15. return (T) mapper.readValue(spec, getSpecClass());
  16. }
  17. }
  18. public static class Spec extends AbstractSpec<Foo> {
  19. public Spec(String spec) {
  20. super(spec);
  21. }
  22. }
  23. @Test
  24. public void test() throws JsonProcessingException {
  25. var foo = new Foo("foo");
  26. var spec = new Spec(mapper.writeValueAsString(foo));
  27. var deserialized = spec.deserializeTo();
  28. Assertions.assertTrue(deserialized instanceof Foo);
  29. }

这里spec类就可以顺利的被反序列化。

这个和最开始失败的case的差别就是新增了一个子类,主要的差别是getGenericSuperclass的返回值有差异,非子类的情况下,获取到的是Object。

因此理论上子类Spec的类型信息中,实际上是保存了父类中的类型参数信息的,也就是例子中的Foo. 按照 https://stackoverflow.com/questions/42874197/getgenericsuperclass-in-java-how-does-it-work 的方式,可以查看到Spec类的字节码中有相应的类型信息。

  1. $ javap -verbose ./org/apache/flink/kubernetes/operator/controller/GenericTest\$Spec.class | grep Signature
  2. #15 = Utf8 Signature
  3. Start Length Slot Name Signature
  4. Signature: #19 // Lorg/apache/flink/kubernetes/operator/controller/GenericTest$AbstractSpec<Lorg/apache/flink/kubernetes/operator/controller/GenericTest$Foo;>;

参考

https://www.cnblogs.com/wuqinglong/p/9456193.html

https://stackoverflow.com/questions/3403909/get-generic-type-of-class-at-runtime

https://stackoverflow.com/questions/6624113/get-type-name-for-generic-parameter-of-generic-class

https://github.com/jhalterman/typetools

Java泛型类型擦除问题的更多相关文章

  1. Java泛型类型擦除以及类型擦除带来的问题

    目录 1.Java泛型的实现方法:类型擦除 1-2.通过两个例子证明Java类型的类型擦除 2.类型擦除后保留的原始类型 3.类型擦除引起的问题及解决方法 3-1.先检查,再编译以及编译的对象和引用传 ...

  2. 记一次由于Java泛型类型擦除而导致的问题,及解决办法

    中所周知,Java中的泛型并不像C++.C#一样是真正的泛型,其泛型是通过类型擦除来实现的.具体什么是类型擦除,可以参看这篇博文:http://icyfenix.iteye.com/blog/1021 ...

  3. Java泛型类型擦除与运行时类型获取

    Java的泛型大家都知道是类型擦除的方式实现的,“编译器会进行泛型擦除”是一个常识了(实际擦除的是参数和自变量的类型).“类型擦除” 并非像许多开发者认为的那样,在 <..> 符号内的东西 ...

  4. Java泛型类型擦除导致的类型转换问题

    初步结论:泛型类型转换不靠谱: 源码: package com.srie.testjava; public class TestClassDefine4<T> { public stati ...

  5. 初探Java类型擦除

    本篇博客主要介绍了Java类型擦除的定义,详细的介绍了类型擦除在Java中所出现的场景. 1. 什么是类型擦除 为了让你们快速的对类型擦除有一个印象,首先举一个很简单也很经典的例子. // 指定泛型为 ...

  6. Java泛型擦除

    Java泛型擦除: 什么是泛型擦除? 首先了解一下什么是泛型?我个人的理解:因为集合中能够存储随意类型的对象.可是集合中最先存储的对象类型一旦确定后,就不能在存储其它类型的对象了,否则,编译时不会报错 ...

  7. Java类型擦除机制

    Java泛型是JDK 5引入的一个特性,它允许我们定义类和接口的时候使用参数类型,泛型在集合框架中被广泛使用.类型擦除是泛型中最让人困惑的部分,本篇文章将阐明什么是类型擦除,以及如何使用它. 一个常见 ...

  8. JAVA泛型-擦除

    package com.xt.thinks15_7; import java.util.Arrays; class EraseObject1<A> { } class EraseObjec ...

  9. JAVA类型擦除

    Java泛型-类型擦除 一.概述 Java泛型在使用过程有诸多的问题,如不存在List<String>.class, List<Integer>不能赋值给List<Num ...

随机推荐

  1. element-ui 无法对绑定表单的对象中的对象属性进行验证

    <el-form-item label="类型" :label-width="formLabelWidth" prop="typeId" ...

  2. Go 语言 结构体

    Go 语言 结构体 引言Go 语言中数组可以存储同一类型的数据,但在结构体中我们可以为不同项定义不同的数据类型结构体是由一系列具有相同类型或不同类型的数据构成的数据集合结构体表示一项记录,比如保存图书 ...

  3. 当心,你搞的Scrum可能是小瀑布

    摘要:有的团队刚接触Scrum,一个问题令他们很困扰:迭代初期开发人员的工作较多,测试人员闲着:迭代末期开发人员闲着,测试人员的工作比较多,怎么解决资源等待的问题呢? 本文分享自华为云社区<当心 ...

  4. HTML5与HTML4区别简介

    移动互联网的快速发展,尤其是4G时代已经来临,加上微软在Windows 10中搭载了新的浏览器Edge取代了IE的地位,所以现在很多网站都开始抛弃IE朝着HTML5发展,PC端在不同浏览器之间的兼容性 ...

  5. css3中user-select的用法详解

    css3中user-select的用法详解 user-select属性是css3新增的属性,用于设置用户是否能够选中文本.可用于除替换元素外的所有元素,以下是user-select的主要用法和注意事项 ...

  6. node-webkit文档翻译#package.json

    title: node-webkit文档翻译#package.json date: 2013-12-07 21:38:25 tags: node-webkit 基本示例 { "main&qu ...

  7. 基于express框架的留言板实现步骤

    这个留言板是基于express框架,和ejs模板引擎,首先需要在根目录安装express框架,然后安装ejs模块和body-parser(获取用户表单提交的数据):建立项目目录 message,然后依 ...

  8. python---冒泡排序的实现

    冒泡排序 思想 ​ 列表中有n个数, 每两个相邻的数, 如果前边的数比后边的数大, 就交换. ​ 关键点: ​ 趟: 总共执行 n-1趟 ​ 无序区: 第 i 趟时, 索引 0~ n-1-i 为无序区 ...

  9. [源码解析] TensorFlow 分布式环境(8) --- 通信机制

    [源码解析] TensorFlow 分布式环境(8) --- 通信机制 目录 [源码解析] TensorFlow 分布式环境(8) --- 通信机制 1. 机制 1.1 消息标识符 1.1.1 定义 ...

  10. Spring集成web环境(手动实现)

    1.创建UserDao接口及其实现类UserDaoImpl(接口代码省略) public class UserDaoImpl implements UserDao { @Override public ...