E和? extends E 以及<? extends E> 和<? super E>

来源:互联网 发布:喜欢下雨天知乎 编辑:程序博客网 时间:2024/04/28 10:55

这是java的泛型语法,E表示泛型具现时使用的类,而? extends E则表示E的子类或者E的实现。

一开始接触的问题是,为什么需要? extends E这样的语法?

例如一个List<Number> numbList的add方法自然可以接受Number的子类,包括Double或者Integer。所以在add的参数只需要是add(E)就可以了,不需要add(? extends E)。

这是因为addAll方法的需要,addAll(Collection<? exntends E>)在这个接口里,如果使用addAll(Collection<E>)是不接受List<Double>作为参数的,这是因为按照官方的说法,java的泛型是“invariant”的。一个List<Double>不是一个List<Number>,确实一个List<? extends Number>。

值得注意的是,数组不一样,一个Double[]是一个Number[]。

参考:http://stackoverflow.com/questions/2770264/what-is-the-difference-between-e-extends-number-and-number


  1. //Apple Orange 都继承自Fruit类,同时Jonathan是Apple的子类   
  2. List<? extends Fruit> list = new ArrayList<Apple>();   
  3. //list.add(new Apple());Error   
  4. //list.add(new Fruit());Error   
  5.   
  6. List<? super Fruit> list = new ArrayList<Apple>();   
  7. list.add(new Apple());//可以   
  8. list.add(new Fruit());//可以   
  9.   
  10. List<? extends Fruit> list = new ArrayList<Apple>();   
  11. list中为什么不能加入Fruit类和Fruit类的子类呢,原因是这样的   
  12.    List<? extends Fruit>表示上限是Fruit,下面这样的赋值都是合法的   
  13.    List<? extends Fruit> list1 = new ArrayList<Fruit>();   
  14.    List<? extends Fruit> list2 = new ArrayList<Apple>();   
  15.    List<? extends Fruit> list3 = new ArrayList<Jonathan>();   
  16.    如果List<? extends Fruit>支持add方法的方法合法的话   
  17.    list1可以add Fruit和所有Fruit的子类   
  18.    list2可以add Apple和所有Apple的子类   
  19.    list3可以add Jonathan和所有Jonathan的子类   
  20.    这样的话,问题就出现了   
  21.    List<? extends Fruit>所应该持有的对象是Fruit的子类,而且具体是哪一个   
  22.    子类还是个未知数,所以加入任何Fruit的子类都会有问题,因为如果add Apple   
  23.    的话,可能List<? extends Fruit>持有的对象是new ArrayList<Jonathan>()   
  24.    Apple的加入肯定是不行的,如果 如果add Jonathan   
  25.    的话,可能List<? extends Fruit>持有的对象是new ArrayList<Jonathan的子类>()   
  26.    Jonathan的加入又不合法,所以List<? extends Fruit> list 不能进行add   
  27.   
  28.       
  29.    而List<? super Fruit> list 表示list持有的对象是Fruit的父类,下限是  Fruit,所以list中add Fruit或Fruit的子类是没有问题的 
0 0
原创粉丝点击