Scala中的协变,逆变,上界,下界等

目录 [−]

Scala中的协变逆变和Java中的协变逆变不一样,看起来更复杂。 本文对Scala中的这些概念做一总结。
首先看几个概念:

  • covariant 协变。使你能够使用比原始指定的类型的子类
  • Contravariance 逆变。使你能够使用比原始指定的类型的父类。
  • Invariance 不变。你只能使用原始指定的类型,不能协变和逆变
  • Upper bounds 上界。
  • Lower bounds 下界。

Java中的协变和逆变

首先我们先回顾一下Java中的协变和逆变,这样我们更容易理解Scala中的协变和逆变。
协变

1
2
3
4
5
6
class Super {
Object getSomething(){}
}
class Sub extends Super {
String getSomething() {}
}

Sub.getSomething()是一个协变类型,因为它的返回类型是Super.getSomething返回类型的子类。

逆变

1
2
3
4
5
6
class Super{
void doSomething(String parameter)
}
class Sub extends Super{
void doSomething(Object parameter)
}

Sub.getSomething()是一个逆变类型,因为它的输入参数是Super.getSomething输入参数的父类。

泛型
泛型中也有协变和逆变。

1
2
3
4
5
6
7
8
9
10
List<String> aList...
List<? extends Object> covariantList = aList;
List<? super String> contravariantList = aList;
 
covariantList.add("d"); //wrong
Object a = covariantList.get(0);
 
contravariantList.add("d"); //OK
String b = contravariantList.get(1); //wrong
Object c = contravariantList.get(2);

你可以调用covariantList所有的不需要泛型参数的方法,因为泛型参数必须 extends Object, 但是编译时你不知道它确切的类型。但是你可以调用getter方法,因为返回类型总是符合Object类型。
contravariantList正好相反,你可以调用所有的带泛型参数的方法,因为你明确的可以传入一个String的父类。但是getter方法却不行。

Scala的协变

首先我们需要了解的是子类型(subtyping)。一个类可以是其它类的子类(sub-)或者父类(super-)。我们可以使用数学概念(partial order)来定义:

1
A -> B iff A <: B

当我们定义一个协变类型List[A+]时,List[Child]可以是List[Parent]的子类型。
当我们定义一个逆变类型List[-A]时,List[Child]可以是List[Parent]的父类型。

看下面的例子:

1
2
3
4
5
6
7
8
9
10
class Animal {}
class Bird extends Animal {}
 
class Consumer[T](t: T) {
 
}
class Test extends App {
val c:Consumer[Bird] = new Consumer[Bird](new Bird)
val c2:Consumer[Animal] = c
}

c不能赋值给c2,因为Consumer定义成不变类型。

稍微改一下:

1
2
3
4
5
6
7
8
9
10
class Animal {}
class Bird extends Animal {}
 
class Consumer[+T](t: T) {
 
}
class Test extends App {
val c:Consumer[Bird] = new Consumer[Bird](new Bird)
val c2:Consumer[Animal] = c
}

因为Consumer定义成协变类型的,所以Consumer[Bird]Consumer[Animal]的子类型,所以它可以被赋值给c2

Scala的逆变

将上面的例子改一下:

1
2
3
4
5
6
7
8
9
10
class Animal {}
class Bird extends Animal {}
 
class Consumer[-T](t: T) {
 
}
class Test extends App {
val c:Consumer[Bird] = new Consumer[Bird](new Bird)
val c2:Consumer[Hummingbird] = c
}

这里Consumer[-T]定义成逆变类型,所以Consumer[Bird]被看作Consumer[Hummingbird]的子类型,故c可以被赋值给c2

下界lower bounds

如果协变类包含带类型参数的方法时:

1
2
3
4
5
6
class Animal {}
class Bird extends Animal {}
 
class Consumer[+T](t: T) {
def use(t: T) = {}
}

编译会出错。出错信息为 "Covariant type T occurs in contravariant position in type T of value t"。
但是如果返回结果为类型参数则没有问题。

1
2
3
4
5
6
class Animal {}
class Bird extends Animal {}
 
class Consumer[+T](t: T) {
def get(): T = {new T}
}

为了在方法的参数中使用类型参数,你需要定义下界:

1
2
3
4
5
6
class Animal {}
class Bird extends Animal {}
 
class Consumer[+T](t: T) {
def use[U >: T](u : U) = {println(u)}
}

上界upper bounds

看一下逆变类中使用上界的例子:

1
2
3
4
5
6
class Animal {}
class Bird extends Animal {}
 
class Consumer[-T](t: T) {
def get[U <: T](): U = {new U}
}

看以看到方法的返回值是协变的位置,方法的参数是逆变的位置。
因此协变类的类型参数可以用在方法的返回值的类型,在方法的参数类型上必须使用下界绑定 >:
逆变类的类型参数可以用在方法的参数类型上,用做方法的返回值类型时必须使用上界绑定 <:

综合协变,逆变,上界,下界

一个综合例子:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
class Animal {}
class Bird extends Animal {}
 
class Consumer[-S,+T]() {
def m1[U >: T](u: U): T = {new T} //协变,下界
def m2[U <: S](s: S): U = {new U} //逆变,上界
}
 
class Test extends App {
val c:Consumer[Animal,Bird] = new Consumer[Animal,Bird]()
val c2:Consumer[Bird,Animal] = c
c2.m1(new Animal)
c2.m2(new Bird)
}

View Bound <%

Scala还有一种视图绑定的功能,如

1
2
3
4
5
6
class Bird {def sing = {}}
class Toy {}
 
class Consumer[T <% Bird]() {
def use(t: T) = t.sing
}

或者类型参数在方法上:

1
2
3
4
5
6
7
8
9
10
11
class Bird {def sing = {}}
class Toy {}
 
class Consumer() {
def use[T <% Bird](t: T) = t.sing
}
 
class Test extends App {
val c = new Consumer()
c.use(new Toy)
}

它要求T必须有一种隐式转换能转换成Bird,也就是 T => Bird,否则上面的代码会编译出错:

1
No implicit view available from Toy => Bird.

加入一个隐式转换,编译通过。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
import scala.language.implicitConversions
 
class Bird {def sing = {}}
class Toy {}
 
class Consumer() {
def use[T <% Bird](t: T) = t.sing
}
 
class Test extends App {
implicit def toy2Bird(t: Toy) = new Bird
val c = new Consumer()
c.use(new Toy)
}

Context Bound

context bound在Scala 2.8.0中引入,也被称作type class pattern
view bound使用A <% String方式,context bound则需要参数化的类型,如Ordered[A]
它声明了一个类型A,隐式地有一个类型B[A],语法如下:

1
def f[A : B](a: A) = g(a) // where g requires an implicit value of type B[A]

更清晰的一个例子:

1
def f[A : ClassManifest](n: Int) = new Array[A](n)

又比如

1
def f[A : Ordering](a: A, b: A) = implicitly[Ordering[A]].compare(a, b)

Scala中的协变,逆变,上界,下界等的更多相关文章

  1. C#中泛型方法与泛型接口 C#泛型接口 List<IAll> arssr = new List<IAll>(); interface IPerson<T> c# List<接口>小技巧 泛型接口协变逆变的几个问题

    http://blog.csdn.net/aladdinty/article/details/3486532 using System; using System.Collections.Generi ...

  2. 解读经典《C#高级编程》最全泛型协变逆变解读 页127-131.章4

    前言 本篇继续讲解泛型.上一篇讲解了泛型类的定义细节.本篇继续讲解泛型接口. 泛型接口 使用泛型可定义接口,即在接口中定义的方法可以带泛型参数.然后由继承接口的类实现泛型方法.用法和继承泛型类基本没有 ...

  3. C#的in/out关键字与协变逆变

    C#提供了一组关键字in&out,在泛型接口和泛型委托中,若不使用关键字修饰类型参数T,则该类型参数是不可变的(即不允许协变/逆变转换),若使用in修饰类型参数T,保证"只将T用于输 ...

  4. Programming In Scala笔记-第十九章、类型参数,协变逆变,上界下界

    本章主要讲Scala中的类型参数化.本章主要分成三个部分,第一部分实现一个函数式队列的数据结构,第二部分实现该结构的内部细节,最后一个部分解释其中的关键知识点.接下来的实例中将该函数式队列命名为Que ...

  5. .NET 4.0中的泛型逆变和协变

    转载自:http://www.cnblogs.com/Ninputer/archive/2008/11/22/generic_covariant.html:自己加了一些理解 随Visual Studi ...

  6. java协变逆变,PECS

    public static void main(String[] args) { // Object <- Fruit <- Apple <- RedApple System.out ...

  7. 协变 & 逆变

    都跟里氏替换原则有关. 协变:你可以用一个子类对象去替换相应的一个父类对象,这是完全符合里氏替换原则的,和协(谐)的变.如:用Swan替换Bird. 逆变:你可以用一个父类对象去替换相应的一个子类对象 ...

  8. C#核心语法讲解-泛型(详细讲解泛型方法、泛型类、泛型接口、泛型约束,了解协变逆变)

    泛型(generic)是C#语言2.0和通用语言运行时(CLR)的一个新特性.泛型为.NET框架引入了类型参数(type parameters)的概念.类型参数使得设计类和方法时,不必确定一个或多个具 ...

  9. C#核心语法-泛型(详细讲解泛型方法、泛型类、泛型接口、泛型约束,了解协变逆变)

    泛型(generic)是C#语言2.0和通用语言运行时(CLR)的一个新特性.泛型为.NET框架引入了类型参数(type parameters)的概念.类型参数使得设计类和方法时,不必确定一个或多个具 ...

随机推荐

  1. C#轻量级日志监控器EasyLogMonitor

    一.课程介绍 本次分享课程属于<C#高级编程实战技能开发宝典课程系列>中的一部分,阿笨后续会计划将实际项目中的一些比较实用的关于C#高级编程的技巧分享出来给大家进行学习,不断的收集.整理和 ...

  2. IntelliJ IDEA2018.1、2017.3破解教程

    (1)下载破解补丁 把下载的破解补丁放在你的idea的安装目录下的bin的目录下面(如下图所示),本文示例为G:\idea\IntelliJ IDEA 2017.3.4 破解补丁下载:http://i ...

  3. UE.getEditor('editor')

    <script type="text/javascript"> $().ready(function(){ var editor = document.getEleme ...

  4. 【T04】开发并使用应用程序框架

    1.TCP/IP应用程序分为 TCP服务器 TCP客户端 UDP服务器 UDP客户端 2.构建框架库是比较简单的一件事,主要就是对socket编程.

  5. mac上配置mysql与redis server,并结合Pydev准备某爬虫环境

    mysql下安装mysql server mysql下安装redis server:https://www.jianshu.com/p/3bdfda703552 mac下安装配置redis:https ...

  6. Jexus 网站服务器和 ASP.NET 跨平台开发

    微软的跨平台战略 微软在过去的一年多中时间中发生了令整个 IT 行业感到惊叹的变化.这一切始于 Ballmer 的退位和 Nadella 的决心,更始于早已在微软各个基层部门蠢蠢欲动的二次创业. 以开 ...

  7. Java Web开发基础零星知识

    1. Web的三个核心标准 万维网的核心标准有三个,分别是URL.HTTP和HTML. URL(统一资源定位符,Universal Resource Locator),为描述Internet上的网页以 ...

  8. Atitit  技术经理职责与流程表总结

    Atitit  技术经理职责与流程表总结 1. (最重要) 理念 价值观建设  ***团队文化建设2 1.1. 加强跨项目组员沟通 ,防止重复劳动2 1.2. 活动聚餐2 2. (重要)方向建设 技术 ...

  9. Understanding How Graal Works - a Java JIT Compiler Written in Java

    https://chrisseaton.com/truffleruby/jokerconf17/ https://chrisseaton.com/truffleruby/tenthings/ http ...

  10. 【iCore4 双核心板_FPGA】例程七:状态机实验——状态机使用

    实验现象:按键每按下一次,三色LED改变一次状态. 核心代码: //--------------------module_rst_n---------------------------// modu ...