上界通配符(? extends Type)
在Java中,? extends Type是一个上界通配符,表示参数化类型的上限是Type。这意味着容器可以持有Type类型的任何对象或者Type的子类型对象。
使用场景
这种类型的通配符常用于泛型方法中,允许方法接受Type的实例或其子类型的集合。这同样基于PECS原则,即如果你需要一个提供(生产)指定类型元素的集合,使用? extends。
例子
假设我们有以下类的层次结构:
class Animal {}
class Dog extends Animal {}
class Cat extends Animal {}
现在,我们有一个方法,它的目的是从一个集合里读取Animal对象:
public void processAnimals(List<? extends Animal> animals) {
for(Animal animal : animals) {
// 这里可以安全地调用Animal类的方法
}
}
这个方法接受的参数是一个列表,这个列表的类型是Animal或Animal的任何子类型。因此,以下的调用是有效的:
List<Animal> animalList = new ArrayList<>();
List<Dog> dogList = new ArrayList<>();
List<Cat> catList = new ArrayList<>();
processAnimals(animalList); // 正确:列表的类型正好是Animal
processAnimals(dogList); // 正确:Dog是Animal的子类型
processAnimals(catList); // 正确:Cat是Animal的子类型
在这个例子中,无论是Animal列表、Dog列表还是Cat列表,都可以传递给processAnimals方法,因为它们都满足? extends Animal的条件。这样做的好处是你可以将方法的使用范围扩展到更多的子类型,同时能够在提取元素时保持类型安全。
遍历
使用? extends Type时,遍历集合是安全的,因为你确切知道集合中的每个元素至少是Type类型。因此,你可以调用Type中定义的方法,而不需要进行任何类型转换。
例如
public void processAnimals(List<? extends Animal> animals) {
for(Animal animal : animals) {
// 可以直接调用Animal类的方法,无需类型转换
}
}
总结
? extends Type通配符的使用提高了代码的灵活性,它允许你编写能够接受更多子类型集合的泛型方法,同时保证了从这些集合中提取元素的类型安全。这是PECS原则中的“Producer Extends”部分,适用于你的集合是提供或生产元素的情况。在遍历这样的集合时,可以直接处理Type类型的元素,无需担心类型转换的问题。