作者热门文章
- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有以下类(class):
case class Box[+A](value: A) {
def set(a: A): Box[A] = Box(a)
}
Error:(4, 11) covariant type A occurs in contravariant position in type A of value a
def set(a: A): Box[A] = Box(a)
最佳答案
一旦你理解它,错误信息实际上非常清楚。让我们一起到达那里。
您正在声明类 Box
作为其类型参数中的协变 A
.这意味着对于任何类型 X
延伸 A
(即 X <: A
)、Box[X]
可以看作是Box[A]
.
为了给出一个明确的例子,让我们考虑 Animal
类型:
sealed abstract class Animal
case class Cat extends Animal
case class Dog extends Animal
如果您定义
Dog <: Animal
和
Cat <: Animal
,然后两者
Box[Dog]
和
Box[Cat]
可以看成
Box[Animal]
你可以例如创建一个包含这两种类型的集合并保留
Box[Animal]
类型。
Box
上提供的操作施加了限制。 .这就是为什么编译器不允许你定义
def set
.
def set(a:A): Unit
那么下面的代码是有效的:
val catBox = new Box[Cat]
val animalBox: Box[Animal] = catBox // valid because `Cat <: Animal`
val dog = new Dog
animalBox.set(dog) // This is non-sensical!
最后一行显然有问题,因为
catBox
现在将包含
Dog
!方法的参数出现在所谓的“逆变位置”中,这与协方差相反。事实上,如果你定义
Box[-A]
,然后
Cat <: Animal
暗示
Box[Cat] >: Box[Animal]
(
Box[Cat]
是
Box[Animal]
的父类(super class)型)。对于我们的示例,这当然是无意义的。
Box
类不可变(即不提供任何方法来更改
Box
的内容),而是使用您的
case class
中定义的 apply 方法创建新盒子的伴侣。如果需要,也可以定义
set
本地而不是将其暴露在任何地方之外
Box
通过将其声明为
private[this]
.编译器将允许这样做,因为
private[this]
保证我们错误示例的最后一行不会编译,因为
set
方法在
Box
的特定实例之外完全不可见.
set
如下。
def set[B >: A](b: B): Box[B] = Box(b)
关于scala - 协变类型 A 出现在值 a 的类型 A 中的逆变位置,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43180310/
COW 不是奶牛,是 Copy-On-Write 的缩写,这是一种是复制但也不完全是复制的技术。 一般来说复制就是创建出完全相同的两份,两份是独立的: 但是,有的时候复制这件事没多大必要
我是一名优秀的程序员,十分优秀!