1.extends
List<? extends Number> foo3
的通配符声明,意味着以下的赋值是合法的:
// Number "extends" Number (in this context)
List<? extends Number> foo3 = new ArrayList<? extends Number>();
// Integer extends Number
List<? extends Number> foo3 = new ArrayList<? extends Integer>();
// Double extends Number
List<? extends Number> foo3 = new ArrayList<? extends Double>();
- 读取操作通过以上给定的赋值语句,你一定能从foo3列表中读取到的元素的类型是什么呢?
- 你可以读取到Number,因为以上的列表要么包含Number元素,要么包含Number的类元素。
你不能保证读取到Integer,因为foo3可能指向的是List。
你不能保证读取到Double,因为foo3可能指向的是List。
- 写入操作过以上给定的赋值语句,你能把一个什么类型的元素合法地插入到foo3中呢?
你不能插入一个Integer元素,因为foo3可能指向List。
你不能插入一个Double元素,因为foo3可能指向List。
你不能插入一个Number元素,因为foo3可能指向List。
你不能往List<? extends T>中插入任何类型的对象,因为你不能保证列表实际指向的类型是什么,你并不能保证列表中实际存储什么类型的对象。唯一可以保证的是,你可以从中读取 到T或者T的子类。
2.super
List<? super Integer> foo3的通配符声明,意味着以下赋值是合法的:
// Integer is a "superclass" of Integer (in this context)
List<? super Integer> foo3 = new ArrayList<Integer>();
// Number is a superclass of Integer
List<? super Integer> foo3 = new ArrayList<Number>();
// Object is a superclass of Integer
List<? super Integer> foo3 = new ArrayList<Object>();
- 读取操作通过以上给定的赋值语句,你一定能从foo3列表中读取到的元素的类型是什么呢?
- 你不能保证读取到Integer,因为foo3可能指向List或者List
你不能保证读取到Number,因为foo3可能指向List。
唯一可以保证的是,你可以读取到Object或者Object子类的对象(你并不知道具体的子类是 什么)。
- 写入操作通过以上给定的赋值语句,你能把一个什么类型的元素合法地插入到foo3中呢?
- 你可以插入Integer对象,因为上述声明的列表都支持Integer。
你可以插入Integer的子类的对象,因为Integer的子类同时也是Integer,原因同上。
你不能插入Double对象,因为foo3可能指向ArrayList。
你不能插入Number对象,因为foo3可能指向ArrayList。
你不能插入Object对象,因为foo3可能指向ArrayList。
3.PECS
请记住PECS原则:生产者(Producer)使用extends,消费者(Consumer)使用super。
- 生产者使用extends
- 如果你需要一个列表提供T类型的元素(即你想从列表中读取T类型的元素),你需要把这个列表声明成<? extends T>,比如List<? extends Integer>,因此你不能往该列表中添加任何元素。
- 消费者使用super
- 如果需要一个列表使用T类型的元素(即你想把T类型的元素加入到列表中),你需要把这个列表 声明成<? super T>,比如List<? super Integer>,因此你不能保证从中读取到的元素的类型。
- 即是生产者,也是消费者
- 如果一个列表即要生产,又要消费,你不能使用泛型通配符声明列表,比如List。
2.上下界通配符的副作用
- 边界让Java不同泛型之间的转换更容易了。但不要忘记,这样的转换也有一定的副作用。那就是容 器的部分功能可能失效。
- 上界<? extends T>不能往里存,只能往外取。
- 下界<? super T>不影响往里存,但往外取只能放在Object对象里