JAVA泛型擦除原理

Java 泛型擦除原理‌

1. 泛型擦除的核心机制

Java 的泛型通过‌类型擦除(Type Erasure) ‌实现,其本质是‌在编译阶段移除泛型类型信息‌,生成的字节码中不保留泛型参数的具体类型‌。

  • 编译时处理 ‌:泛型类型参数(如 <T>)会被替换为‌边界类型 ‌(默认 Object,或用户指定的上限类型,如 <T extends Number> 替换为 Number)‌。
  • 运行时无泛型信息 ‌:例如,List<String>List<Integer> 在编译后均为 List,其 Class 对象相同(ArrayList.class)‌。
2. 擦除过程的具体表现
  • 类型参数替换 ‌:

    泛型类/方法中的类型参数会被替换为具体边界类型。例如:

    java 复制代码
    // 编译前
    public class Box<T> {
        private T data;
        public void set(T data) { this.data = data; }
    }
    
    // 编译后(类型擦除后)
    public class Box {
        private Object data;
        public void set(Object data) { this.data = data; }
    }

    若泛型有上限(如 <T extends Number>),则替换为 Number‌。

  • 强制类型转换插入 ‌:

    从泛型对象中获取数据时,编译器自动插入类型转换代码。例如:

    java 复制代码
    List<String> list = new ArrayList<>();
    String s = list.get(0); // 编译后实际代码:String s = (String) list.get(0);
3. 泛型擦除的设计原因
  • 向后兼容性 ‌:Java 1.5 引入泛型时需保证旧版本非泛型代码(如 ArrayList)仍能运行,避免修改 JVM 底层机制‌。
  • 简化 JVM 实现‌:类型擦除使 JVM 无需为泛型生成新字节码,沿用原有类加载机制‌。
4. 类型擦除的局限性
  • 运行时类型信息丢失‌:无法通过反射获取泛型参数的具体类型(如无法判断 List<String> 的 String)‌。

  • 类型约束失效场景 ‌:
    通过反射可绕过泛型限制,例如向 List<Integer> 插入 String 对象(编译时不报错,运行时报错)‌。

    java 复制代码
    List<Integer> list = new ArrayList<>();
    list.add(1);
    list.getClass().getMethod("add", Object.class).invoke(list, "abc"); // 运行时成功插入
    Integer num = list.get(1); // 抛出 ClassCastException
5. 泛型擦除的验证示例
java 复制代码
List<String> strList = new ArrayList<>();
List<Integer> intList = new ArrayList<>();
System.out.println(strList.getClass() == intList.getClass()); // 输出 true‌:ml-citation{ref="1,5" data="citationList"}

总结

Java 泛型擦除通过‌编译时类型替换 ‌和‌自动插入类型转换 ‌实现泛型功能,核心目的是‌兼容旧版本代码 ‌和‌简化 JVM 设计‌。虽然牺牲了运行时类型信息,但通过编译器检查保证了类型安全‌。

相关推荐
华仔啊39 分钟前
主线程存了用户信息,子线程居然拿不到?ThreadLocal 背锅
java·后端
间彧1 小时前
Spring Boot项目中,Redis 如何同时执行多条命令
java·redis
召摇1 小时前
如何避免写垃圾代码:Java篇
java·后端·代码规范
vker1 小时前
第 1 天:单例模式(Singleton Pattern)—— 创建型模式
java·设计模式
我不是混子1 小时前
什么是内存泄漏?
java
程序员小假1 小时前
我们来说说当一个线程两次调用 start() 方法会出现什么情况?
java·后端
SimonKing2 小时前
Archery:开源、一站式的数据库 SQL 审核与运维平台
java·后端·程序员
皮皮林55114 小时前
IDEA 源码阅读利器,你居然还不会?
java·intellij idea
卡尔特斯18 小时前
Android Kotlin 项目代理配置【详细步骤(可选)】
android·java·kotlin
白鲸开源18 小时前
Ubuntu 22 下 DolphinScheduler 3.x 伪集群部署实录
java·ubuntu·开源