什么是 PE,CS
PE,CS全称是producer extends,consumer super的缩写,这是 Joshua Bloch 在 Effective Java 一书 中引入的一个略显奇怪的术语,但有助于理解泛型的用法。换言之,参数化类型代表 生产者(producer)则使用 extends, 代表消费者(consumer)则使用 super
PE 原则
简单来说, PE 表示,如果你的方法只是想从集合获取值,并且希望集合的类型范围是T及其子类,那么泛型可以定义为
? extends T
了解泛型的同学应该知道 extends
是上界通配符,使用了上界通配符,只能读取值,不能写入值,取值的的类型是T
只能取值不能写入的就是生产者,有人可能有点理解了,我们来个更详细的解释,举个栗子
假设,我们有个水果的对象,水果类有个addAll
方法,用来将另一个水果集合,放入到水果对象内
class Fruit{
private List<Fruit> fruits = new LinkedList<>();
public void addAll(List<Fruit> fruits){
for (Fruit fruit : fruits) {
this.fruits.add(fruit);
}
}
}
现在,我们需求变了,有苹果,香蕉分别继承了水果这个类,如果我么将苹果或者香蕉集合直接放入addAll
方法,会编译报错,因为苹果集合,不是水果集合
public static void main(String[] args){
List<Apple> appleList = new LinkedList<>();
Fruit fruit = new Fruit();
// 编译报错
fruit.addAll(appleList);
}
class Apple extends Fruit{
}
class Banana extends Fruit{
}
如果我们要放进去怎么办,使用上界通配符,修改addAll
方法,使用了上界通配符后,元素只能读,不能写,传入的集合类型范围是Fruit或者其子类集合
public void addAll(List<? extends Fruit> f){
for (Fruit fruit : f) {
this.fruits.add(fruit);
}
有人会问了,这个上界通配符,跟PE有什么关系? 当然有,PE是producer extends
的缩写,我们仔细看看,对于addAll
方法来说,只是要消费入参,那么入参就是生产者,说着这,估计很多人已经明白了
PE是针对方法来说的,如果某个方法的入参,需要一个生产者,并且范围是泛型的子类,那么使用上界通配符extends
CS 原则
简单来说, CS 表示,如果你的方法只是想往集合写入值,并且集合的类型范围希望是T及其父类,那么泛型可以定义为
? super T
来表示
了解泛型的同学应该知道,super
表示下界通配符,使用了下界通配符,只能写入值,不能取值,写入的值必须是T或者其子类
只能写入,不能取出就是消费者,有人可能有点理解了,我们来个更详细的解释,举个例子
我们还是沿用上面的例子,我们进行改造一下,苹果类有个addAll
方法,传入一个苹果集合,将苹果对象放入这个集合里面
public class Apple extends Fruit{
private Apple apple = new Apple();
public void addAll(List<Apple> apples){
apples.add(apple);
}
}
现在需求变了,我们变更,我们想传入一个更大的水果集合,将苹果添加入水果集合里面,毕竟苹果是属于水果的子类,我们直接将水果集合传入进去,会编译报错,我们改造一下addAll
方法,使用下界通配符将它的接收范围扩大,传入的集合范围是Apple或者其父类集合
public void addAll(List<? super Apple> apples){
apples.add(apple);
}
有人会问了,这个下界通配符,跟CS有什么关系? 当然有,CS是consumer super
的缩写,对于addAll
方法来说,只要是将东西放进到入参内,那么入参就是消费者
CS也是针对方法来说的,如果某个方法的入参需要消费方法内的东西,入参并且是泛型的父类,那么使用下界通配符super
总结
如果进行简单的归纳,那就是
- 只从方法入参集合获取值,那么使用 extends
- 只从方法入参集合写入值,那么使用 super