深入剖析Java泛型

作者: Louis_陆 | 来源:发表于2016-05-17 21:14 被阅读487次

    首先,我们要从概念上认识泛型,泛型的作用是什么?泛型是为了能够在编译时而不是在运行时检测出错误而产生的,以提高程序的可靠性、健壮性和鲁棒性。

    有部分读者可能会有这样的理解误区:
    如定义泛型ArrayList<String>list,比定义原始类型ArrayList list,能提高程序运行的效率,能减少JVM底层识别数据类型的时间。
    注意!不管实际的具体类型是什么,泛型类是被它的所有实例所共享的,在JVM中只有原始类型被存储为单独一个类

    如下面的例子所示

    ArrayList<String> list1 = new ArrayList<String>();
    ArrayList<Integer> list2 = new ArrayList<Integer>();
    System.out.println(list1 instanceof ArrayList);  //true
    System.out.println(list2 instanceof ArrayList);  //true
    System.out.println(list1 instanceof ArrayList<String>);  //false
    

    由此可知,泛型存在于编译时,一旦编译器确认泛型类型是安全使用的,就会将它转换为原始类型,JVM中只含有原始类型,所以上面的理解是错误的,泛型对运行时是毫无意义的,更谈不上提高效率。

    正由于泛型的这个特点,所以不能用泛型类型参数创建实例
    E object = new E(); // false

    此外,不能用泛型类型参数创建数组
    E[] elements = new E [capacity]; //false

    同时不建议使用强制类型转换
    E[] elements = (E[])new Object [capacity];
    如果 E 是 String,而 new Object[] 是 Integer 对象的数组,那么 (String[]) (new Object[]) 将会导致 ClassCastException 异常。

    在静态环境下、异常类也是不能使用泛型的。

    下面我们来观察一个例子:

    import java.util.ArrayList;
    import java.util.List;
    
    class A {
    }
    
    class B extends A {
    }
    
    class D extends B {
    }
    
    public class TestGenericType {
        public static void main(String[] args) {
    
            List list = new ArrayList();
            List<A> listA = new ArrayList<A>();
            list = listA;// A正确
    
            List<B> listB = new ArrayList<B>();
            // listA=listB;//B错误
    
            List<Object> listObject = new ArrayList<Object>();
            List<?> list$ = new ArrayList<>();
            list$ = listObject;// C正确
    
            List<D> listD = new ArrayList<D>();
            List<? extends B> listExtendsB = new ArrayList<>();
            listExtendsB = listD;// D正确
    
            List<? extends A> listExtendsA = new ArrayList<>();
            // listA = listExtendsA;//E错误
    
            listExtendsA = listExtendsB;// F正确
    
            List<? super B> listSuperB = new ArrayList<>();
            listSuperB = listB;// G正确
    
            // listSuperB = listExtendsB;//H错误
        }
    
    }
    

    其中, ? 、 ? extends T 、 ? super T 是通配泛型。
    ? --->非受限通配,它和 ? extends Object 一样
    ? extends T --->受限通配,表示 T 或 T 的一个未知子类型
    ? super T --->下限通配,表示 T 或 T 的一个未知父类型

    它们的关系如下图(摘自《Java语言程序设计》进阶篇)

    Paste_Image.png

    我们来做下分析:
    【A】由第一张图的例子可知,listA instanceof list
    【B】尽管 B 是 A 的子类,但 List<B> 不是 List<A> 的子类,所以 listA = listB 是错误的
    【C】? 同等于 ? extends Object,list$ 能接收 Object 或 Object 的一个未知子类型,所以能接收 listObject
    【D】解释同“C”
    【E】同“A”,若 listExtendsA 接收的是 B 类,则 List<B> 不是 List<A> 的子类
    【F】listExtendsA 能接收 A 或 A 的子类,正确
    【G】listSuperB 能接收 B 或 B 的父类,正确
    【H】listSuperB 不能接收 B 的子类,listExtendsB 可能接收 D类

    相关文章

      网友评论

      本文标题:深入剖析Java泛型

      本文链接:https://www.haomeiwen.com/subject/kevirttx.html