泛型中? super T和? extends T的区别

转载自并发编程网 – ifeve.com****
经常发现有List、Set的声明,是什么意思呢?
super T>表示包括T在内的任何T的父类,
T>表示包括T在内的任何T的子类,下面我们详细分析一下两种通配符具体的区别。
extends
List foo3的通配符声明,意味着以下的赋值是合法的:

01 // Number "extends" Number (in this context)’
02
03 List foo3 =newArrayList();
04
05 // Integer extends Number
06
07 List foo3 =newArrayList();
08
09 // Double extends Number
10
11 List foo3 =newArrayList();

读取操作通过以上给定的赋值语句,你一定能从foo3列表中读取到的元素的类型是什么呢?你可以读取到Number,因为以上的列表要么包含Number元素,要么包含Number的类元素。
你不能保证读取到Integer,因为foo3可能指向的是List。
你不能保证读取到Double,因为foo3可能指向的是List。
写入操作过以上给定的赋值语句,你能把一个什么类型的元素合法地插入到foo3中呢?
你不能插入一个Integer元素,因为foo3可能指向List。
你不能插入一个Double元素,因为foo3可能指向List。
你不能插入一个Number元素,因为foo3可能指向List。
你不能往List中插入任何类型的对象,因为你不能保证列表实际指向的类型是什么,你并不能保证列表中实际存储什么类型的对象。唯一可以保证的是,你可以从中读取到T或者T的子类。
super
现在考虑一下List。
List foo3的通配符声明,意味着以下赋值是合法的:

01 // Integer is a "superclass" of Integer (in this context)
02
03 List foo3 =newArrayList();
04
05 // Number is a superclass of Integer
06
07 List foo3 =newArrayList();
08
09 // Object is a superclass of Integer
10
11 List foo3 =newArrayList();

读取操作通过以上给定的赋值语句,你一定能从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。
PECS
请记住PECS原则:生产者(Producer)使用extends,消费者(Consumer)使用super。
生产者使用extends
如果你需要一个列表提供T类型的元素(即你想从列表中读取T类型的元素),你需要把这个列表声明成,比如List,因此你不能往该列表中添加任何元素。
消费者使用super
如果需要一个列表使用T类型的元素(即你想把T类型的元素加入到列表中),你需要把这个列表声明成,比如List,因此你不能保证从中读取到的元素的类型。
即是生产者,也是消费者
如果一个列表即要生产,又要消费,你不能使用泛型通配符声明列表,比如List。
例子
请参考java.util.Collections里的copy方法(JDK1.7):

我们可以从Java开发团队的代码中获得到一些启发,copy方法中使用到了PECS原则,实现了对参数的保护。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,854评论 18 139
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,742评论 18 399
  • Collection ├List │├LinkedList │├ArrayList │└Vector │└Stac...
    AndyZX阅读 891评论 0 1
  • 亲子日记第五十二篇:初战告捷 心心念的期中考试成绩终于出炉了! 卢恩萱大中午的跑回来,一边掏口袋一...
    64fdf7a8ac71阅读 283评论 0 0