原文:
http://stackoverflow.com/questions/4343202/difference-between-super-t-and-extends-t-in-java
看原文没怎么看懂 ,加入了自己的一些理解,记录一下
首先下面的例子能通过编译这是前提。
extends:
List<? extends Number> foo1 = new ArrayList<Number>(); // Number "extends" Number (in this context)
List<? extends Number> foo2 = new ArrayList<Integer>(); // Integer extends Number
List<? extends Number> foo3 = new ArrayList<Double>(); // Double extends Number
读:
foo1一定能读到Number类型
foo2不一定能读到Integer类型,比如存的是Double 因为Double也是继承至Number
foo3不一定能读到Double类型,比如存的是Integer类型,因为Integer也是继承至Number
写:
。。。见译文:http://ifeve.com/difference-between-super-t-and-extends-t-in-java/
读:
foo1一定能读到Number类型
foo2不一定能读到Integer类型,比如存的是Double 因为Double也是继承至Number
foo3不一定能读到Double类型,比如存的是Integer类型,因为Integer也是继承至Number
写:
。。。见译文:http://ifeve.com/difference-between-super-t-and-extends-t-in-java/
因为foo2 foo3 具体是存的 integer double 所以你往里面
写的时候 不能写其他例如Float 虽然
Float也是继承至Number,
super:
List<? super Integer> foo4 = new ArrayList<Integer>(); // Integer is a "superclass" of Integer (in this context)
List<? super Integer> foo5 = new ArrayList<Number>(); // Number is a superclass of Integer
List<? super Integer> foo6 = new ArrayList<Object>(); // Object is a superclass of Integer
因为foo5 foo6 具体实例类型为Number Object ,所以你
读的时候 不一定能读到Integer ,虽然Number Object都是Integer的超类
所以原文作者用的PECS原则
Read 读数据的时候 Extends
Write 写数据的时候Supper
最后对于 有可能同时需要读写的时候,请参考java.util.Collections里的copy方法(JDK1.7):
所以原文作者用的PECS原则
请记住PECS原则:生产者(Producer)使用extends,消费者(Consumer)使用super。
- 生产者使用extends
如果你需要一个列表提供T类型的元素(即你想从列表中读取T类型的元素),你需要把这个列表声明成<? extends T>,比如List<? extends Integer>,因此你不能往该列表中添加任何元素。
- 消费者使用super
如果需要一个列表使用T类型的元素(即你想把T类型的元素加入到列表中),你需要把这个列表声明成<? super T>,比如List<? super Integer>,因此你不能保证从中读取到的元素的类型。
- 即是生产者,也是消费者
如果一个列表即要生产,又要消费,你不能使用泛型通配符声明列表,比如List<Integer>。
但是我自己方便记忆 取了个名字 REWSRead 读数据的时候 Extends
Write 写数据的时候Supper
最后对于 有可能同时需要读写的时候,请参考java.util.Collections里的copy方法(JDK1.7):