常常发现有List<? super T>、Set<? extends T>的声明,是什么意思呢?<? super T>表示包括T在内的任何T的父类,<? extends T>表示包括T在内的任何T的子类,下面咱们详细分析一下两种通配符具体的区别。java
List<? extends Number> foo3的通配符声明,意味着如下的赋值是合法的:this
// 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>。你 不能保证读取到Double,由于foo3可能指向的是List<Integer>。spa
写入操做过以上给定的赋值语句,你能把一个什么类型的元素合法地插入到foo3中呢?你不能插入一个Integer元素,由于foo3可能指向 List<Double>。你不能插入一个Double元素,由于foo3可能指向List<Integer>。你不能插入一个 Number元素,由于foo3可能指向List<Integer>。你不能往List<? extends T>中插入任何类型的对象,由于你不能保证列表实际指向的类型是什么,你并不能保证列表中实际存储什么类型的对象。惟一能够保证的是,你能够从中读 取到T或者T的子类。code
如今考虑一下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>();
读取操做经过以上给定的赋值语句,你必定能从foo3列表中读取到的元素的类型是什么呢?你不能保证读取到Integer,由于foo3可能指向 List<Number>或者List<Object>。你不能保证读取到Number,由于foo3可能指向 List<Object>。惟一能够保证的是,你能够读取到Object或者Object子类的对象(你并不知道具体的子类是什么)。io
写入操做经过以上给定的赋值语句,你能把一个什么类型的元素合法地插入到foo3中呢?你能够插入Integer对象,由于上述声明的列表都支持 Integer。你能够插入Integer的子类的对象,由于Integer的子类同时也是Integer,缘由同上。你不能插入Double对象,由于 foo3可能指向ArrayList<Integer>。你不能插入Number对象,由于foo3可能指向 ArrayList<Integer>。你不能插入Object对象,由于foo3可能指向 ArrayList<Integer>。class
请记住PECS原则:生产者(Producer)使用extends,消费者(Consumer)使用super。泛型
生产者使用extendsList
若是你须要一个列表提供T类型的元素(即你想从列表中读取T类型的元素),你须要把这个列表声明成<? extends T>,好比List<? extends Integer>,所以你不能往该列表中添加任何元素。
消费者使用super
若是须要一个列表使用T类型的元素(即你想把T类型的元素加入到列表中),你须要把这个列表声明成<? super T>,好比List<? super Integer>,所以你不能保证从中读取到的元素的类型。
便是生产者,也是消费者
若是一个列表即要生产,又要消费,你不能使用泛型通配符声明列表,好比List<Integer>。
请参考java.util.Collections里的copy方法(JDK1.7):
咱们能够从Java开发团队的代码中得到到一些启发,copy方法中使用到了PECS原则,实现了对参数的保护。