为什么我们不能从逆变类型中读取

Sun*_*vel 3 java generics contravariance

我正在学习 Java 中的泛型,我正在学习协方差和逆变。我了解协方差,为什么我们不能写入协方差类型。但对我来说,逆变是令人困惑的。我想知道为什么从逆变读取总是对象类型

假设我们有以下课程

Class Animal
Class Dog extends Animal
Class Tiger extends Animal
Run Code Online (Sandbox Code Playgroud)

现在,考虑以下代码

List<Animal> animals = new ArrayList<>();
List<? super Animal> contraVarianceList=animals;

animals.add(new Dog()); // Allowed
animals.add(new Tiger()); //Allowed
Animal animal = contraVarianceList.get(0); // Not allowed. Why?
Run Code Online (Sandbox Code Playgroud)

我的问题是为什么我们不能从逆变列表中读取“动物”。为什么总是而且只有“Object”可以返回?如果我们从逆变列表中读取“Animal”,将会或可能会出现什么问题?

jbx*_*jbx 5

我想你是想问为什么Animal animal=contraVarianceList.get(0);不允许。

这样想:

List<Animal> animals=  new ArrayList<>();
List<Object> objects = new ArrayList<>();

List<? super Animal> contraVarianceList = animals;
List<? super Animal> contraVarianceList2 = objects;

animals.add(new Dog()); // Allowed
animals.add(new Tiger()); //Allowed

objects.add(new Dog()); // Allowed
objects.add(new Tiger()); //Allowed

//there is no type difference between contraVarianceList and contraVarianceList2
Animal animal = contraVarianceList.get(0);  //compiler will complain
Animal animal2 = contraVarianceList2.get(0); //compiler will complain
Run Code Online (Sandbox Code Playgroud)

编译器无法真正知道它们中的任何一个只携带 Animal 实例。您明确表示? super Animal,它也可能是 List 携带普通Object.

通常,您将这些类型的列表用于希望添加项目而不是从列表中读取的方法的参数。所以有这样的事情:

void populateAnimals(List<? super Animal> animals) {
  //whatever code
  animals.add(...);
}
Run Code Online (Sandbox Code Playgroud)

保证您可以将方法与List<Object>和 一起使用List<Animal>