Scala 中的协变,逆变和非变

  协变、逆变、非变介绍
协变和逆变主要是用来解决参数化类型的泛化问题。Scala 的协变与逆变是非常有特色
的,完全解决了Java中泛型的一大缺憾;举例来说,Java中,如果有 A 是 B 的子类,但 Card[A]
却不是 Card[B] 的子类;而 Scala 中,只要灵活使用协变与逆变,就可以解决此类 Java 泛
型问题;
由于参数化类型的参数(参数类型)是可变的,当两个参数化类型的参数是继承关系(可
泛化),那被参数化的类型是否也可以泛化呢?Java 中这种情况下是不可泛化的,然而 Scala
提供了三个选择,即协变(“+”)、逆变(“-”)和非变。
下面说一下三种情况的含义,首先假设有参数化特征 Queue,那它可以有如下三种定义。
(1) trait Queue[T] {}
这是非变情况。这种情况下,当类型 B 是类型 A 的子类型,则 Queue[B]与 Queue[A]没
有任何从属关系,这种情况是和 Java 一样的。
(2) trait Queue[+T] {}
这是协变情况。这种情况下,当类型 B 是类型 A 的子类型,则 Queue[B]也可以认为是
Queue[A]的子类型,即 Queue[B]可以泛化为 Queue[A]。也就是被参数化类型的泛化方向与
参数类型的方向是一致的,所以称为协变。
(3) trait Queue[-T] {}
这是逆变情况。这种情况下,当类型 B 是类型 A 的子类型,则 Queue[A]反过来可以认
为是 Queue[B]的子类型。也就是被参数化类型的泛化方向与参数类型的方向是相反的,所
以称为逆变。

 C[+T]:如果 A 是 B 的子类,那么 C[A]是 C[B]的子类。
 C[-T]:如果 A 是 B 的子类,那么 C[B]是 C[A]的子类。
 C[T]: 无论 A 和 B 是什么关系,C[A]和 C[B]没有从属关系

 

package drt.mavenscala.scala.classdemo

class Super
class Sub extends Super
//协变
class Temp1[+A](title: String)
//逆变
class Temp2[-A](title: String)
//非变
class Temp3[A](title: String)
object Covariance_demo{
  def main(args: Array[String]) {
    //支持协变 Temp1[Sub]还是 Temp1[Super]的子类
    val t1: Temp1[Super] = new Temp1[Sub]("hello scala!!!")
    //支持逆变 Temp1[Super]是 Temp1[Sub]的子类
    val t2: Temp2[Sub] = new Temp2[Super]("hello scala!!!")
    //支持非变 Temp3[Super]与 Temp3[Sub]没有从属关系,如下代码会报错
    //val t3: Temp3[Sub] = new Temp3[Super]("hello scala!!!")
    //val t4: Temp3[Super] = new Temp3[Sub]("hello scala!!!")
    println(t1.toString)
    println(t2.toString)
  }
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值