术语
英文 | 中文 | 示例 |
---|---|---|
Variance | 型变 | Function[-T, +R] |
Nonvariant | 不变 | Array[A] |
Covariant | 协变 | Supplier[+A] |
Contravariant | 逆变 | Consumer[-A] |
Immutable | 不可变 | String |
Mutable | 可变 | StringBuilder |
其中,Mutable常常意味着Nonvariant,但是Noncovariant(不可协变的)与Mutable分别表示两个不同的范畴。
即:可变的,一般意味着“不可型变”,但是“不可协变”和可变的,分别表示两个不同范畴。
型变(Variance)拥有三种基本形态:协变(Covariant), 逆变(Contravariant), 不变(Nonconviant),可以形式化地描述为:
一般地,假设类型C[T]持有类型参数T;给定两个类型A和B,如果满足A <: B,则C[A]与 C[B]之间存在三种关系
如果C[A] <: C[B],那么C是协变的(Covariant);
如果C[A] :> C[B],那么C是逆变的(Contravariant);
否则,C是不变的(Nonvariant)。
Scala的类型参数使用+标识“协变”,-标识“逆变”,而不带任何标识的表示“不变”(Nonvariable)
trait C[+A] // C is covariant
trait C[-A] // C is contravariant
trait C[A] // C is nonvariant
如何判断一个类型是否有型变能力
一般地,“不可变的”(Immutable)类型意味着“型变”(Variant),而“可变的”(Mutable)意味着“不变”(Nonvariant)。
其中,对于不可变的(Immutable)类型C[T]
如果它是一个生产者,其类型参数应该是协变的,即C[+T]。
如果它是一个消费者,其类型参数应该是逆变的,即C[-T] 。
协变和逆变点