上一篇文章中我们介绍了泛型的基础知识点,详情请参考文章:
今天我们来继续讲解泛型中另一个非常重要的概念,就是那个“小问号”——通配符!
通配符概念
泛型中除了用
- public class Car{
- public void drive() {
- System.out.println("car的drive方法");
-
- };
- public void brake() {
- System.out.println("car的brake方法");
- };
- }
- public class Benz extends Car{
- public void drive() {
- System.out.println("benz drive");
- };
- }
根据上面的代码,因为Benz 是Car的子类,所以 Car c=new Benz();是成立的,那么,ArrayList
所以,我们得出结论:Benz 和Car有继承关系,不代表 List< Benz >和 List
通配符有 3 种形式。
- >被称作无限定的通配符。
- extends T>被称作有上限的通配符。
- super T>被称作有下限的通配符。
无限定通配符 >经常与容器类配合使用,它其中的 ? 其实代表的是未知类型,所以涉及到 ? 的操作,一定与具体类型无关。这里extends 和super与泛型上下边界中的extends 和super的概念是一致的,由于在前面的文章中介绍过,这里也就不再赘述了。 extends T> 解决了这样一个问题,代码如下所示:
从上面的代码中我们可以看到:
extends Car>虽然可以编译通过,但是l2也不能往里存,我们只能调用与类型无关的操作方法,代码如下:
我们可以看到使用add方法直接报错了!因为?是未知的。但是调用如下方法是没有问题的。
- l2.get(0);
- l2.size();
- l2.iterator().next();
这里大家要了解 super T>的特殊性,它具有一定程度的写操作的能力,代码如下:
- ArrayList super Benz> l3= new ArrayList<>();
- l3.add(new Benz()); //成功
- l3.add(new Car()); //编译不通过
所以,提供了只读的功能,也就是它删减了增加具体类型元素的能力,只保留与具体类型无关的功能。它不管装载在这个容器内的元素是什么类型,它只关心元素的数量以及容器是否为空。
T和?的主要区别
最后我们总结一下T和?的主要区别:
区别一
T代表确定的类型,这里的确定是指运行时确定。
?代表未知类型,所以它涉及的操作都基本上与类型无关,因此 jvm 不需要针对它对类型作判断,因此它能编译通过
区别二
通过T来确保泛型参数的一致性,下面这两个参数的类型是一致的
- public
Void test(List p1, List p2)
通配符是不确定的,所以下面这个方法不能保证两List具有相同的元素类型
- public void test(List extends Number>p1,List Extends Number>p2)
区别三
Class
- // 可以
- public Class> clazz;
-
- // 不可以,因为 T 需要指定类型
- public Class
clazzT;