泛型(四)之泛型擦除

2020-05-05  本文已影响0人  极客天空

一 、Java泛型的原理?什么是泛型擦除机制?

Java的泛型是JDK5新引入的特性,为了向下兼容,虚拟机其实是不支持泛型,所以Java实现的是一种伪泛型机制,也就是说Java在编译期擦除了所有的泛型信息,这样Java就不需要产生新的类型到字节码,所有的泛型类型最终都是一种原始类型,在Java运行时根本就不存在泛型信息。

二、Java编译器具体是如何擦除泛型的

  1. 检查泛型类型,获取目标类型
  2. 擦除类型变量,并替换为限定类型

如果泛型类型的类型变量没有限定(<T>),则用Object作为原始类型
如果有限定(<T extends XClass>),则用XClass作为原始类型
如果有多个限定(T extends XClass1&XClass2),则使用第一个边界XClass1作为原始类

  1. 在必要时插入类型转换以保持类型安全
  2. 生成桥方法以在扩展时保持多态性

三、使用泛型以及泛型擦除带来的影响(副作用)

  1. 泛型类型变量不能使用基本数据类型
    比如没有ArrayList<int>,只有ArrayList<Integer>.当类型擦除后,
    ArrayList的原始类中的类型变量(T)替换成Object,但Object类型不能
    存放int值
  2. 不能使用instanceof 运算符
    因为擦除后,ArrayList<String>只剩下原始类型,泛型信息String
    不存在了,所有没法使用instanceof


    instenceof1.png
  3. 泛型在静态方法和静态类中的问题
    因为泛型类中的泛型参数的实例化是在定义泛型类型对象
    (比如ArrayList<Integer>)的时候指定的,而静态成员是不需要使用
    对象来调用的,所有对象都没创建,如何确定这个泛型参数是什么


    静态类型.png
  4. 泛型类型中的方法冲突
    因为擦除后两个equals方法变成一样的了
    @Override
    public boolean equals(T t) {
        return super.equals(t);
    }

    @Override
    public boolean equals(Object o) {
        return super.equals(o);
    }
  1. 没法创建泛型实例
    因为类型不确定


    泛型不能创建实例.png
  2. 没有泛型数组
    因为数组是协变,擦除后就没法满足数组协变的原则
    逆变与协变
上一篇 下一篇

猜你喜欢

热点阅读