? 通配符类型安全
<? extends T> 表示类型的上界,表示参数化类型的多是T 或是 T的子类app
<? super T> 表示类型下界(Java Core中叫超类型限定),表示参数化类型是此类型的超类型(父类型),直至Objectui
static class Food{} static class Fruit extends Food{} static class Apple extends Fruit{} static class RedApple extends Apple{} List<? extends Fruit> flist = new ArrayList<Apple>(); // complie error: // flist.add(new Apple()); // flist.add(new Fruit()); // flist.add(new Object()); flist.add(null); // only work for null
List<? extends Frut> 表示 “具备任何从Fruit继承类型的列表”,编译器没法肯定List所持有的类型,因此没法安全的向其中添加对象。能够添加null,由于null 能够表示任何类型。因此List 的add 方法不能添加任何有意义的元素,可是能够接受现有的子类型List<Apple> 赋值。spa
Fruit fruit = flist.get(0); Apple apple = (Apple)flist.get(0);
因为,其中放置是从Fruit中继承的类型,因此能够安全地取出Fruit类型。code
flist.contains(new Fruit()); flist.contains(new Apple());
在使用Collection中的contains 方法时,接受Object 参数类型,能够不涉及任何通配符,编译器也容许这么调用。orm
List<? super Fruit> flist = new ArrayList<Fruit>(); flist.add(new Fruit()); flist.add(new Apple()); flist.add(new RedApple()); // compile error: List<? super Fruit> flist = new ArrayList<Apple>();
List<? super Fruit> 表示“具备任何Fruit超类型的列表”,列表的类型至少是一个 Fruit 类型,所以能够安全的向其中添加Fruit 及其子类型。因为List<? super Fruit>中的类型多是任何Fruit 的超类型,没法赋值为Fruit的子类型Apple的List<Apple>.对象
// compile error: Fruit item = flist.get(0);
由于,List<? super Fruit>中的类型多是任何Fruit 的超类型,因此编译器没法肯定get返回的对象类型是Fruit,仍是Fruit的父类Food 或 Object.继承
extends 可用于的返回类型限定,不能用于参数类型限定。 super 可用于参数类型限定,不能用于返回类型限定。 >带有super超类型限定的通配符能够向泛型对易用写入,带有extends子类型限定的通配符能够向泛型对象读取。——《Core Java》