深入剖析Java泛型

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

有部分读者可能会有这样的理解误区:
如定义泛型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类

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Scala与Java的关系 Scala与Java的关系是非常紧密的!! 因为Scala是基于Java虚拟机,也就是...
    灯火gg阅读 3,505评论 1 24
  • 开发人员在使用泛型的时候,很容易根据自己的直觉而犯一些错误。比如一个方法如果接收List作为形式参数,那么如果尝试...
    时待吾阅读 1,074评论 0 3
  • 为什么需要泛型? 通过泛型可以定义类型安全的数据结构,而无须使用实际的数据类型(可扩展)。这能够显著提高性能并得到...
    一只好奇的茂阅读 1,279评论 2 39
  • 我好像多读了两章,今天就给自己一个懒的理由,这两天一直在读前世今生,不知道为啥就是对这个特别感兴趣,继续看!
    小怪兽_d0c7阅读 167评论 0 0
  • 我总是觉得美好得事物值得被等待,殊不知,等待却蹉跎了最好得时光。 也许有人会说你眼高于顶,认为你好高骛远,或者觉得...
    隐沐在人间阅读 640评论 0 1