Java类型擦除是编译期将泛型参数替换为边界类型(如Object、Number等)并移除泛型信息的过程,旨在兼容旧JVM;编译器自动插入类型转换保证安全,但导致instanceof、泛型数组、new T()等受限,并通过桥接方法解决多态问题。

Java中的类型擦除,是指编译器在编译阶段把泛型类型参数(如 Object,有上界时则用上界类型),最终生成的字节码里不保留任何泛型信息。它不是运行时行为,而是编译期的“翻译动作”,目的是让泛型代码能与 Java 5 之前的旧 JVM 和类库无缝共存。
编译器按以下逻辑处理泛型参数:
T)→ 替换为 Object
T extends Number)→ 替换为 Number
T extends A & B & C)→ 替换为第一个类型 A(因 Java 接口继承链限制)<t> void foo(T t)</t>)→ 擦除为 void foo(Object t)
虽然运行时没了泛型信息,但编译器会在调用点自动插入强制类型转换,把返回值“补回”你声明的类型:
List<string> list = new ArrayList();</string>String s = list.get(0); → 编译后实际等价于:String s = (String) list.get(0);
ClassCastException
这些不是 bug,而是擦除机制的自然结果,必须在编码时主动规避:
立即学习“Java免费学习笔记(深入)”;
if (obj instanceof List<string>)</string> 直接编译失败,只能写 if (obj instanceof List)
new ArrayList<string>[10]</string> 编译报错;可行方案是先建 Object[10] 再转型或用 List<list>></list>
T 在运行时不存在;需传入 Class<t></t> 并用反射构造,例如 clazz.getDeclaredConstructor().newInstance()
void handle(List<string>)</string> 和 void handle(List<integer>)</integer> 擦除后都是 handle(List),编译报错当子类覆写父类泛型方法并指定具体类型时,编译器会自动生成一个“桥接方法”来衔接擦除后的签名差异。例如:
class Parent<t> { void set(T t) {} }</t>
class Child extends Parent<string> { @Override void set(String s) {} }</string>
void set(Object o) { set((String) o); },确保多态调用仍能正确分发以上就是在Java中类型擦除是怎么回事_Java泛型擦除机制解析的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号