无法使用通配符泛型类型为Java集合添加值

Ser*_*nov 25 java generics

为什么这段代码不能编译(Parent是一个接口)?

List<? extends Parent> list = ...
Parent p = factory.get();   // returns concrete implementation
list.set(0, p);   // fails here: set(int, ? extends Parent) cannot be applied to (int, Parent)
Run Code Online (Sandbox Code Playgroud)

Jon*_*eet 45

为了安全起见,这样做.想象一下,如果它有效:

List<Child> childList = new ArrayList<Child>();
childList.add(new Child());

List<? extends Parent> parentList = childList;
parentList.set(0, new Parent());

Child child = childList.get(0); // No! It's not a child! Type safety is broken...
Run Code Online (Sandbox Code Playgroud)

意思List<? extends Parent>是"这是一个扩展的某种类型的列表Parent.我们不知道哪种类型 - 它可能是a List<Parent>,a List<Child>或a List<GrandChild>." 这使得它的安全抓取任何物品出来的的List<T>API和转换TParent,但它不是安全地调用List<T>API从转换ParentT......因为转换可能是无效的.

  • 我认为多态性的坏例子呢?"孩子"不是(总是)"父母". (3认同)
  • @justhalf:“Parent”在问题中,并表明“Child”是子类的自然示例。不是我从头开始选择的类名,但就问题所在的上下文而言足够清晰。 (2认同)

Boz*_*zho 16

List<? super Parent>
Run Code Online (Sandbox Code Playgroud)

PECS - "制作人 - 扩展,消费者 - 超级".你ListParent对象的消费者.