【Java基础面试题034】Java泛型擦除是什么?

news/2024/12/26 2:20:22/

回答重点

泛型擦除指的是Java编译器在编译时将所有泛型信息删除的过程,以确保与Java1.4及之前的版本保持兼容

泛型参数在运行时会被替换为其上界(通常是Object),这样一来在运行时无法获取泛型的实际类型。

作用:泛型擦除确保了Java代码的向下兼容性,即可以与旧版本的Java代码兼容,为了让使用泛型的代码在不同版本的Java运行时环境中都可以正常工作。又由于泛型类型信息在编译时期被擦除了,因此运行时无法获取泛型信息,这样就不能创建泛型类型的数组或对泛型类型使用instanceof检查

示例:

java">public <T> void printList(List<T> list) {for (T element : list) {System.out.println(element);}
}

编译后的代码类似于:

java">public void printList(List list) {for (Object element : list) {System.out.println(element);}
}

类型T会被擦除成Object

扩展知识

为什么Java泛型的实现是类型擦除?

回答重点提到了主要原因是为了向下兼容,即兼容Java5之前编译的class文件。

例如Java1.2上正在跑的代码,可以在Java1.5上的JRE上运行

也是因为需要向下兼容,才使得Java实现的是伪泛型

我从现有的实现倒推伪泛型的设计可能思路(个人瞎掰的,做个参考即可)

是这样的:

  1. 这Java5以前的版本在线上已经有很多应用在跑了,如果我的Java5不能兼容,没人用啊
  2. 泛型毕竟是加一个约束,以前的代码没有这个约束啊,该如何兼容?
  3. 有了,要不我在编译器上动手动脚,在编译的时候识别和约束泛型,然后编译过了就把泛型的信息擦除了。这样运行的时候就没有约束了,跟之前的版本就没什么区别了。

参考网上的解释:

这说明,写Java的也是程序员,也是要发版有上线需求的

为什么运行期通过反射可以获得类型?

java">public class GenericTest {public static void main(String[] args) {List<String> list = new ArrayList<>();list.add("hello");String s = list.get(0);}
}

看字节码 

从反编译看生成的字节码文件能看到,new的List没有保存泛型,所以是被擦除了

下面有一步checkcast强转为String,可是Java代码中不需要写强转呢?

因为编译器隐形的帮我们插入了强转的代码所以不需要我们写

再看标题:既然擦除了类型,为什么运行期还能通过反射获取类型?

答案就藏在class文件中

看下面代码:

获取泛型类型

java">public class GenericTest {public List<String> list;public static void main(String[] args) throws Exception{Field field = GenericTest.class.getField("list");Type type = ((ParameterizedType) field.getGenericType()).getActualTypeArguments()[0];System.out.println(type);}
}

javap -v,可以看到字节码里记录了泛型类型信息,所以编译器虽然擦除了泛型类型,也记录了泛型信息,自然能通过反射获取

由于静态记录在了字节码中,所以局部变量这种存在栈中的泛型类型,通过反射就无法获取

只有三种情况可以通过反射获取泛型类型:

  • 成员变量的泛型
  • 方法入参的泛型
  • 方法返回值的泛型
  • 类的泛型


http://www.ppmy.cn/news/1558147.html

相关文章

python中functools 起什么作用

在 Python 中&#xff0c;functools 是一个标准库模块&#xff0c;提供了一组实用工具&#xff0c;用于操作函数和可调用对象。它常用于装饰器、函数缓存、部分应用等&#xff0c;帮助开发者编写更简洁和高效的代码。 functools 的作用和常用功能 以下是 functools 模块中几个…

功能篇:JAVA8实现数据去重

在Java 8中&#xff0c;有多种方法可以实现集合的去重。下面我将介绍几种常见的方法&#xff1a; ### 使用Set接口 最简单的方法是使用Set接口&#xff0c;因为根据定义&#xff0c;Set不允许重复元素。如果你有一个List并且想要去除其中的重复项&#xff0c;你可以将其转换为…

蓝桥杯摆烂第三天

小蓝给学生们组织了一场考试&#xff0c;卷面总分为 100 分&#xff0c;每个学生的得分都是一个 0 到 100 的整数。 请计算这次考试的最高分、最低分和平均分。 输入描述 输入的第一行包含一个整数 n (1≤n≤104)&#xff0c;表示考试人数。 接下来 n 行&#xff0c;每行包…

【C++】sophus : sim_details.hpp 实现了矩阵函数 W、其导数,以及其逆 (十七)

这段代码主要用于计算与旋转矩阵和指数相关的矩阵运算&#xff0c;是Sophus库的一部分&#xff0c;Sophus是用于几何运算的C库。以下是对代码的总结&#xff1a; 主要功能 calcW: 计算矩阵W&#xff0c;该矩阵与旋转矩阵和指数有关。根据不同的theta和sigma值&#xff0c;进行特…

CPU性能优化-磁盘空间和解析时间

即使考虑了跟踪文件的压缩格式&#xff0c;编码后的数据仍然会占用很大的磁盘空间。通常&#xff0c;每条指令不超过1字节&#xff0c;但是考虑到CPU执行指令的速度&#xff0c;数据仍然非常多。根据负载&#xff0c;CPU编码以100MB/s的速度处理PT跟踪文件的情况是很常见的&…

【蓝桥杯每日一题】分糖果——DFS

分糖果 蓝桥杯每日一题 2024-12-24 分糖果 DFS 题目描述 两种糖果分别有 9 个和 16 个&#xff0c;要全部分给 7 个小朋友&#xff0c;每个小朋友得到的糖果总数最少为 2 个最多为 5 个&#xff0c;问有多少种不同的分法。糖果必须全部分完。 只要有其中一个小朋友在两种方案中…

重温设计模式--代理模式

文章目录 定义UML图代理模式主要有以下几种常见类型&#xff1a;代理模式涉及的主要角色有&#xff1a;C 代码示例 定义 代理模式&#xff08;Proxy Pattern&#xff09;属于结构型设计模式&#xff0c;它为其他对象提供一种代理以控制对这个对象的访问。 通过引入代理对象&am…

Java复习|图形用户界面AWT、Swing----银行客户管理系统【校课版】【1】

校课总结&#xff0c;部分&#xff0c;未完待续...... 背景了解 Java的AWT和Swing的现状 AWT&#xff08;Abstract Window Toolkit&#xff09; AWT是Java中最早期的图形用户界面&#xff08;GUI&#xff09;工具包&#xff0c;它直接与操作系统提供的图形函数进行交互&a…