1

我理解参数化集合,如果你想使用参数化类型的子类型,你需要将集合声明为Collection<? extends Whatever>

例如:

public interface Fruit {}
public interface Banana extends Fruit {}

void thisWorksFine() {
  //Collection<Fruit> fruits;          //wrong
  Collection<? extends Fruit> fruits;  //right
  Collection<Banana> bananas = new ArrayList<>();
  fruits = bananas;
}

但是如果我添加一个额外的层,这会爆炸:

public interface Box<T> {}

void thisDoesNotCompile() {
    Collection<Box<? extends Fruit>> boxes;
    Collection<Box<Banana>> bananaBoxes = new ArrayList<>();
    boxes = bananaBoxes;  // error!
}

出现错误:

error: incompatible types
required: Collection<Box<? extends Fruit>>
found:    Collection<Box<Banana>>

为什么这些不兼容?有什么办法可以让它工作吗?

4

1 回答 1

4

因为你可以添加一个Box<Apple>to boxes,这会违反bananaBoxes.

public interface Apple extends Fruit {}

//...

Box<Apple> apples = new Box<>(); // this is legal
Box<? extends Fruit> fruits = apples; // this is legal

Collection<Box<Banana>> bananaBoxes = new ArrayList<>(); 

Collection<Box<? extends Fruit>> boxes = bananaBoxes; //if this were legal...
boxes.add(fruits); //then this would be legal

//and this would be a type violation:
Box<Banana> bananas = bananaBoxes.iterator().next(); 

你可以改为

Collection<? extends Box<? extends Fruit>> boxes = bananaBoxes;

这是合法的,因为它可以防止上述情况。

于 2012-10-22T18:47:36.290 回答