原文出处: 并发编程网java
常常发现有List<? super T>、Set<? extends T>的声明,是什么意思呢?<? super T>表示包括T在内的任何T的父类,<? extends T>表示包括T在内的任何T的子类,下面咱们详细分析一下两种通配符具体的区别。编程
List<? extends Number> foo3的通配符声明,意味着如下的赋值是合法的:并发
01 |
// Number "extends" Number (in this context) |
02 |
03 |
List<? extends Number> foo3 = new ArrayList<? extends Number>(); |
04 |
05 |
// Integer extends Number |
06 |
07 |
List<? extends Number> foo3 = new ArrayList<? extends Integer>(); |
08 |
09 |
// Double extends Number |
10 |
11 |
List<? extends Number> foo3 = new ArrayList<? extends Double>(); |
你不能保证读取到Integer,由于foo3可能指向的是List<Double>。ide
你不能保证读取到Double,由于foo3可能指向的是List<Integer>。函数
你不能插入一个Integer元素,由于foo3可能指向List<Double>。this
你不能插入一个Double元素,由于foo3可能指向List<Integer>。spa
你不能插入一个Number元素,由于foo3可能指向List<Integer>。.net
你不能往List<? extends T>中插入任何类型的对象,由于你不能保证列表实际指向的类型是什么,你并不能保证列表中实际存储什么类型的对象。惟一能够保证的是,你能够从中读取到T或者T的子类。code
如今考虑一下List<? super T>。对象
List<? super Integer> foo3的通配符声明,意味着如下赋值是合法的:
01 |
// Integer is a "superclass" of Integer (in this context) |
02 |
03 |
List<? super Integer> foo3 = new ArrayList<Integer>(); |
04 |
05 |
// Number is a superclass of Integer |
06 |
07 |
List<? super Integer> foo3 = new ArrayList<Number>(); |
08 |
09 |
// Object is a superclass of Integer |
10 |
11 |
List<? super Integer> foo3 = new ArrayList<Object>(); |
你不能保证读取到Number,由于foo3可能指向List<Object>。
惟一能够保证的是,你能够读取到Object或者Object子类的对象(你并不知道具体的子类是什么)。
你能够插入Integer的子类的对象,由于Integer的子类同时也是Integer,缘由同上。
你不能插入Double对象,由于foo3可能指向ArrayList<Integer>。
你不能插入Number对象,由于foo3可能指向ArrayList<Integer>。
你不能插入Object对象,由于foo3可能指向ArrayList<Integer>。
请记住PECS原则:生产者(Producer)使用extends,消费者(Consumer)使用super。
若是你须要一个列表提供T类型的元素(即你想从列表中读取T类型的元素),你须要把这个列表声明成<? extends T>,好比List<? extends Integer>,所以你不能往该列表中添加任何元素。
若是须要一个列表使用T类型的元素(即你想把T类型的元素加入到列表中),你须要把这个列表声明成<? super T>,好比List<? super Integer>,所以你不能保证从中读取到的元素的类型。
若是一个列表即要生产,又要消费,你不能使用泛型通配符声明列表,好比List<Integer>。
请参考java.util.Collections里的copy方法(JDK1.7):
引用例子:
泛型中使用通配符有两种形式:子类型限定<? extends xxx>和超类型限定<? super xxx>。
下面的代码定义了一个Pair<T>类,以及Employee,Manager和President类。
如今要定义一个函数能够打印Pair<Employee>
但是有一个问题是这个函数输入参数只能传递类型Pair<Employee>,而不能传递Pair<Manager>和Pair<President>。例以下面的代码会产生编译错误
之因此会产生编译错误,是由于Pair<Employee>和Pair<Manager>其实是两种类型。
由上图能够看出,类型Pair<Manager>是类型Pair<? extends Employee>的子类型,因此为了解决这个问题能够把函数定义改为
public static void printEmployeeBoddies(Pair<? extends Employee> pair)
可是使用通配符会不会致使经过Pair<? extends Employee>的引用破坏Pair<Manager>对象呢?例如:
Pair<? extends Employee> employeePair = managePair;employeePair.setFirst(new Employee("Tony", 100));
不用担忧,编译器会产生一个编译错误。Pair<? extends Employee>参数替换后,咱们获得以下代码
? extends Employee getFirst()
void setFirst(? extends Employee)
对于get方法,没问题,由于编译器知道能够把返回对象转换为一个Employee类型。可是对于set方法,编译器没法知道具体的类型,因此会拒绝这个调用。
超类型限定和子类型限定相反,能够给方法提供参数,可是不能使用返回值。? super Manager这个类型限定为Manager的全部超类。
Pair<? super Manager>参数替换后,获得以下方法
? super Manager getFirst()
void setFirst(? super Manager)
编译器能够用Manager的超类型,例如Employee,Object来调用setFirst方法,可是没法调用getFirst,由于不能把Manager的超类引用转换为Manager引用。
超类型限定的存在是为了解决下面一类的问题。例如要写一个函数从Manager[]中找出工资最高和最低的两个,放在一个Pair中返回。
如此就能够这样调用