其他分享
首页 > 其他分享> > 泛型中 <? super T> 和 <? extends T> 的区别

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

作者:互联网

原文链接: https://ifeve.com/difference-between-super-t-and-extends-t-in-java/

经常发现有List<? super T>、Set<? extends T>的声明,是什么意思呢?<? super T>表示包括T在内的任何T的父类,<? extends T>表示包括T在内的任何T的子类,下面我们详细分析一下两种通配符具体的区别。

extends

List<? extends Number> foo3的通配符声明,意味着以下的赋值是合法的:

// Number "extends" Number (in this context)

List<? extends Number> foo3 = new ArrayList<Number>(); 

// Integer extends Number

List<? extends Number> foo3 = new ArrayList<Integer>();

// Double extends Number

List<? extends Number> foo3 = new ArrayList<Double>();

你不能往List<? extends T>中插入任何类型的对象,因为你不能保证列表实际指向的类型是什么,你并不能保证列表中实际存储什么类型的对象。唯一可以保证的是,你可以从中读取到T或者T的子类。

super

现在考虑一下List<? super T>。

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>();

PECS

请记住PECS原则:生产者(Producer)使用extends,消费者(Consumer)使用super。

生产者使用extends
如果你需要一个列表提供T类型的元素(即你想从列表中读取T类型的元素),你需要把这个列表声明成<? extends T>,比如List<? extends Integer>,因此你不能往该列表中添加任何元素。

消费者使用super
如果需要一个列表使用T类型的元素(即你想把T类型的元素加入到列表中),你需要把这个列表声明成<? super T>,比如List<? super Integer>,因此你不能保证从中读取到的元素的类型。

即是生产者,也是消费者
如果一个列表即要生产,又要消费,你不能使用泛型通配符声明列表,比如List。

例子
请参考java.util.Collections里的copy方法(JDK1.7):
在这里插入图片描述
我们可以从Java开发团队的代码中获得到一些启发,copy方法中使用到了PECS原则,实现了对参数的保护。

标签:读取,List,Number,列表,extends,泛型,Integer,super,foo3
来源: https://blog.csdn.net/sxudong2010/article/details/110732680