从Joshua Bloch的Effective Java中,
>
协变简单地说,如果X是Y的子型,那么X[]也将是Y[]的子型。数组是协变的,因为字符串是对象的子类型,所以
String[]是Object[]的子类型
不变简单地说,不管X是不是Y的子类型,
List<X> will not be subType of List<Y>.
我的问题是为什么决定在Java中使数组是协变的?还有其他的SO帖子,比如为什么数组是不变的,但是列表是协变的?,但它们似乎集中在Scala上,我无法理解。
通过维基百科:
Java和C#的早期版本不包含泛型(也称为参数多态)。
在这样的设置中,使数组不变就排除了有用的多态程序。例如,考虑编写一个函数来洗牌一个数组,或者编写一个函数来使用元素上的object.equals
方法来测试两个数组是否相等。该实现不依赖于存储在数组中的元素的确切类型,因此应该可以编写一个在所有类型的数组上工作的单个函数。类型的函数很容易实现
boolean equalArrays (Object[] a1, Object[] a2);
void shuffleArray(Object[] a);
但是,如果数组类型被视为不变的,则只能对类型正好为object[]
的数组调用这些函数。例如,不能洗牌字符串数组。
因此,Java和C#都是协变地对待数组类型的。例如,在C#中,String[]
是Object[]
的子类型,而在Java中,String[]
是Object[]
的子类型。
这回答了“为什么数组是协变的?”的问题,或者更准确地说,“为什么当时数组是协变的?”
当泛型被引入时,由于Jon Skeet的回答中指出的原因,它们被故意地不变为协变的:
不,列表
不是列表
。考虑一下您可以对列表
执行什么操作-您可以向其中添加任何动物...包括一只猫。现在,你能合乎逻辑地在一窝小狗身上加上一只猫吗?绝对不行.
// Illegal code - because otherwise life would be Bad
List<Dog> dogs = new List<Dog>();
List<Animal> animals = dogs; // Awooga awooga
animals.add(new Cat());
Dog dog = dogs.get(0); // This should be safe, right?
突然你有一只迷糊的猫。
wikipedia文章中描述的使数组协变的最初动机并不适用于泛型,因为通配符使协变(和逆变)的表达成为可能,例如:
boolean equalLists(List<?> l1, List<?> l2);
void shuffleList(List<?> l);
问题内容: 摘自Joshua Bloch的Effective Java, 数组在两个重要方面不同于通用类型。第一数组是协变的。泛型是不变的。 协变量仅表示如果X是Y的子类型,则X []也将是Y []的子类型。数组是协变的,因为字符串是Object的子类型,所以 不变式仅表示X是否为Y的子类型, 问题答案: Java和C#的早期版本不包含泛型(又称参数多态性)。 在这种情况下,使数组不变会排除有用的
从《狗的名单》中我了解到,《狗的名单》也是一份与直觉完美吻合的动物名单。从定义::[B 我从数组中了解到,狗的数组不是(不能代替a)动物的数组,这是相当违反直觉的。一组狗确实也是一组动物,但显然斯卡拉不同意。 我希望有人能直观地解释为什么数组是不变量的,最好是用狗(或猫)来解释。 这就是为什么数组是不变的,而列表是协变的?但我正在寻找一个更直观的解释,它不(严重)涉及类型系统。 与Scala的不可
这可能是一个很傻的问题,但我挠头了很久也弄不明白其中的区别。 我正在浏览scala泛型页面:https://docs.scala-lang.org/tour/generic-classes.html 注意:泛型类型的子类型是不变的。这意味着,如果我们有一个stack[Char]类型的字符堆栈,那么它就不能用作stack[Int]类型的整数堆栈。这是不合理的,因为它使我们能够将真整数输入到字符堆栈中
在爪哇,its说: 所以数组被称为协变的。但对于泛型,他们说: 因此它是不变的。但问题是,“泛型真的是不变的吗”? 那为什么说泛型是不变的呢?
问题内容: Java中的协变返回类型是什么?在一般的面向对象编程中? 问题答案: 协变返回,意味着当一个方法被覆盖时,覆盖方法的返回类型被允许为覆盖方法的返回类型的子类型。 为了举例说明,通常情况是-声明为返回类型。你可以在自己的类中重写此方法,如下所示: 这样做的好处是,任何持有对MyFoo对象的显式引用的方法都将能够调用clone()并知道(无需强制转换)返回值是的实例。如果没有协变量返回类型
编译以下代码失败: 如: 但如果我这样做: