博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Java 泛型: 什么是PECS(Producer Extends, Consumer Super)
阅读量:6543 次
发布时间:2019-06-24

本文共 1691 字,大约阅读时间需要 5 分钟。

  hot3.png

什么是PECS? 

PECS指“Producer Extends,Consumer Super”。换句话说,如果参数化类型表示一个生产者,就使用<? extends T>;如果它表示一个消费者,就使用<? super T>,可能你还不明白,不过没关系,接着往下看好了。

下面是一个简单的Stack的API接口:

1
2
3
4
5
6
public    
class     
Stack<E>{
        
public    
Stack();
        
public    
void    
push(E e):
        
public    
E pop();
        
public    
boolean    
isEmpty();
}

假设想增加一个方法,按顺序将一系列元素全部放入Stack中,你可能想到的实现方式如下:

1
2
3
4
public    
void    
pushAll(Iterable<E> src){
        
for    
(E e : src)
            
push(e)
}

假设有个Stack<Number>,想要灵活的处理Integer,Long等Number的子类型的集合

1
2
3
Stack<Number> numberStack =    
new    
Stack<Number>();
Iterable<Integer> integers = ....;
numberStack.pushAll(integers);

此时代码编译无法通过,因为对于类型Number和Integer来说,虽然后者是Number的子类,但是对于任意Number集合(如List<Number>)不是Integer集合(如List<Integer>)的超类,因为泛型是不可变的。

幸好java提供了一种叫有限通配符的参数化类型,pushAll参数替换为“E的某个子类型的Iterable接口”:

1
2
3
4
public    
void    
pushAll(Iterable<?    
extends    
E> src){
        
for    
(E e: src)
            
push(e);
}

这样就可以正确编译了,这里的<? extends E>就是所谓的 producer-extends。这里的Iterable就是生产者,要使用<? extends E>。因为Iterable<? extends E>可以容纳任何E的子类。在执行操作时,可迭代对象的每个元素都可以当作是E来操作。

与之对应的是:假设有一个方法popAll()方法,从Stack集合中弹出每个元素,添加到指定集合中去。

1
2
3
4
5
public    
void    
popAll(Collection<E> dst){
           
if    
(!isEmpty()){
                    
dst.add(pop());
            
}
}

假设有一个Stack<Number>和Collection<Object>对象:

1
2
3
Stack<Number> numberStack =    
new    
Stack<Number>();
Collection<Object> objects = ...;
numberStack.popAll(objects);

同样上面这段代码也无法通过,解决的办法就是使用Collection<? super E>。这里的objects是消费者,因为是添加元素到objects集合中去。使用Collection<? super E>后,无论objects是什么类型的集合,满足一点的是他是E的超类,所以不管这个参数化类型具体是什么类型都能将E装进objects集合中去。

总结:

  1. 如果你是想遍历collection,并对每一项元素操作时,此时这个集合时生产者(生产元素),应该使用 Collection<? extends Thing>.

  2. 如果你是想添加元素到collection中去,那么此时集合时消费者(消费元素)应该使用Collection<? super Thing>

注:此文根据《》以及 整理成文。想了解更多有关泛型相关知识,请读者阅读《Effective Java》的第五章。

 

转载于:https://my.oschina.net/sub/blog/201113

你可能感兴趣的文章
主库 归档 删除策略
查看>>
路过下载攻击利用旧版 Android 漏洞安装勒索软件
查看>>
ThinkSNS 六大子版本体验及源码下载
查看>>
《算法基础》——1.5实际因素
查看>>
《Java数字图像处理:编程技巧与应用实践》——第3章 基本Swing UI组件与图像显示 3.1 JPanel组件与BufferedImage对象的显示...
查看>>
为什么有人讨厌 Google 的新 Logo?
查看>>
腾讯2017暑期实习编程题3
查看>>
Intellij IDEA 构建Spring Web项目 — 用户登录功能
查看>>
[AHOI2013]作业
查看>>
git push被忽略的文件 处理
查看>>
C#中用ILMerge将所有引用的DLL打成一个DLL文件
查看>>
使用makecontext实现用户线程【转】
查看>>
Comet:基于 HTTP 长连接的“服务器推”技术
查看>>
BZOJ 2733: [HNOI2012]永无乡 启发式合并treap
查看>>
四种方法校验数组中是否包含某个指定的字符串
查看>>
安装OpenResty开发环境
查看>>
第0课 从0开始
查看>>
hadoop无法启动DataNode问题
查看>>
java泛型中<?>和<T>区别
查看>>
这里是指推送通知跟NSNotification有区别:
查看>>