• 首页 首页 icon
  • 工具库 工具库 icon
    • IP查询 IP查询 icon
  • 内容库 内容库 icon
    • 快讯库 快讯库 icon
    • 精品库 精品库 icon
    • 问答库 问答库 icon
  • 更多 更多 icon
    • 服务条款 服务条款 icon

协变 (Covariant)、逆变 (Contravariant) 和不变 (Invariant)

武飞扬头像
小麦先生
帮助1

1. 定义

协变与逆变 (Covariance and contravariance) 是在计算机科学中,描述具有父/子型别关系的多个型别通过型别构造器、构造出的多个复杂型别之间是否有父/子型别关系的用语。

协变、逆变和不变,属于变型 (variance) 的三种结果,变型指类型构造器如何根据内部类型关系组成自己的类型关系。

协变与逆变这个概念并不局限于某种语言,支持继承与多态的语言都会遇到这个概念。

  • 协变(covariant),如果它保持了子类型序关系≦。该序关系是:子类型≦基类型。
  • 逆变(contravariant),如果它逆转了子类型序关系。
  • 不变(invariant),如果上述两种均不适用。

2. 例子

对于任意类型关系而言,子类型可以胜任父类型的任何场景。(里氏替换原则

对于简单类型关系 Animal 和 Cat 而言,Cat 是 Animal 的子类型。那么对于复杂类型构造器:

  • IEnumerable<> 是协变的,因为 IEnumerable<Cat> 总是 IEnumerable<Animal> 的子类:

    在一个需要 IEnumerable<Animal> 的地方,主调方对迭代器的操作总是希望返回一个 Animal,而将 IEnumerable<Cat> 当成 IEnumerable<Animal> 用,则会返回一个 Cat,Cat 可以胜任 Animal 的任何场景。所以说:类型构造器对其内部类型只有抛出操作时,类型构造器通常是协变的。

  • Action<> 是逆变的,因为 Action<Animal> 总是 Action<Cat> 的子类:

    在一个需要 Action<Cat> 的地方,主调方对一个 Action<Cat> 的调用,总是希望传进一个 Cat 时,操作顺利进行,而将 Action<Animal> 当成 Action<Cat> 用,则只要传进一个 Animal 即可顺利进行,实参传 Cat 在任何场景下都合理。所以说:类型构造器对其内部类型只有接收操作时,类型构造器通常是逆变的。(关于这个例子的疑问见第 5 节)

  • IList<> 是不变的,因为 IList<Cat> 既不能当 IList<Animal> 的子类,也不能当它的父类:

    如果 IList<Cat> 是 IList<Animal> 的子类,那么当主调方拥有一个 IList<Animal>(但它实际是 IList<Cat>)且想把一个 Dog 塞进去时,明明是合法操作,但操作却不安全。

    如果 IList<Animal> 是 IList<Cat> 的子类,那么当主调方拥有一个 IList<Cat>(但实际是 IList<Animal>)且想从中得到一个 Cat 时,有可能得到了一个 Dog,操作不安全。

    但如果列表是只读 (read-only) 的,那么 IList<> 就可以是协变的。(附注:只读数据类型称为源 (Source) ,只写数据类型称为汇 (Sink))。

所以说:类型构造器对其内部类型既有抛出操作,又有接收操作时,类型构造器应该是不变的。

类型构造符→对输入类型是逆变的对输出类型是协变的。这一规则首先被Luca Cardelli正式提出。[1]

3. 有什么用?

为什么我们需要讨论协变、逆变和不变?在设计编程语言,或是进阶使用编程语言的数组、继承和泛型时,必须将变型列入考量,否则可能会违反类型安全,影响程序运行时的健壮性。

正如例子中所表述的,迭代器是协变的,那么内部类型越具体,迭代器需要做的就越多(迭代器这个例子可能不太明显,子类实现只需要比父类实现多一个类型转换);函数是逆变的,即要求的参数越泛化,函数需要做的就越多;列表是不变的,那么任意两种元素类型的列表,都应该当成不同的类型对待

考虑这样一个例子:一个对数组排序的 sort 函数,如果需要适用于各种类型的数组,那么 sort 函数要求的参数(形参)就需要足够泛化,这使得 sort 需要做非常多繁琐的事情,甚至繁琐到无法实现。比如说 sort 接收一个 Object 数组,这足够泛化,但函数内部并不知道数组每个元素实际是什么类型,那么比较方法就无法确定,毕竟 int 也不能跟 string 去比较大小。由于考虑入参的函数是逆变的,这使得函数的泛化性扩展受到了约束。

关于协变的例子就比较直观,协变是很容易理解的,协变的类型构造器的特化性扩展受到自然约束。要支持越具体的类,就需要做各样的判断和特例操作。譬如,用一个 Food<> 类来生产粮食,以喂养动物,Food<Animal> 太过泛化,需要生产所有动物都能吃的粮食,所有动物都能吃的或许只有水了,我们 Food<Animal> 只能生产水,猫狗喝了都没事;而对于 Food<Cat> 猫食构造机,那么除了水,还能生产鱼干,对于 Food<Dog> 狗食构造机,除了水还能生产骨头,且不应生产巧克力等等。由于特化本身就是同细胞分裂一样无限延伸的,所以特化性扩展自然受到约束。

这两个例子说明了什么呢?1. [逆变]依赖于某个组件的,组件越特化,行为越好写;2. [协变]服务于某个组件的,组件越泛化,行为越好写。 这种指导思想在我们设计框架或是组件时,可以帮助我们拿捏泛化和特化的程度(当然这里说的特化是指更具体化的意思,并不是 C 模板编程中的特化 (template specialization),虽然核心思想也差不多)。

而不变 (invariant) 呢,在设计类型构造器时,不变的类型构造器是很影响复用性的,等于说强行要分类讨论,每种类型具体分析,要去复用或许还需要分析行为本身有没有共同点,然后去抽取公共函数。不变性或许很影响复用性,但对类型系统而言是最简单的情况,无论它实际是逆变的还是协变的,当成不变永远不会出错。这里就不多讨论了,可以查看后面的参考文献。

4. 应对

我们更希望从中总结出能够应对逆变或是协变特性的编程思想或范式,不过这件事情或许应该交给更专业的作者来做。出于记录和分享的目的,我仍然在这一小节给出了自己的观点。

  • 逆变阻止了类型构造器无限泛化,对于越泛化的依赖项,类型构造器能获得的特性支持越少。这种情况下,有两种方式可以解决:1. 构建约束,使依赖项有基本的特性保证; 2. 依赖插槽(依赖注入),插入实现构造器所需求的特性实现。很多时候可能两种方式需要一起使用,特别是第一种,约束在很多情况下是必须的。

    还是 sort 函数的例子,我们需要约束传入数组的元素类型是统一的,否则问题的规模过于庞大(无限种类型和无限种类型之间的大小比较)。然后,约束数组元素实现比较运算符(或实现某种可比较接口),或是让 sort 函数接收一个 compare 比较函数插槽,让插槽去处理泛化所抹去的必要特性。

    在各大语言 (C#, Java, etc.) 中,使用泛型 (Generics) 去做这种类型约束,而使用插槽拓展 sort 函数的灵活性。注意使用泛型之后,不同类型数组被当成不变的,由具体类型去生成具体的 sort 行为,接收不同数组的 sort 函数之间也不存在类型关系。

  • 协变的类型构造器无法无限具体化,服务对象越具体化,类型构造器能从父类中得到的帮助越少。这种情况下,应该1. 制定好类型构造器的行为边界,以便父类实现能更大程度地为子类服务(复用);2. 可以将行为中的差异化操作委托给服务对象实现,以减少类型构造器的设计冗余。

    第 1 点不举例子,关于第 2 点,在 Food<> 例子中,我们可以为 Animal 类设计一个 bool isEdible(food) 函数接口,让动物自己告诉食物机,某个食物能不能吃。这样一来,Food<> 只需要关注如何生产粮食,并调用 isEdible 筛选菜单即可,借由泛型/模板,可以使 Food<> 不断特化下去。

5. 其他问题

  • 逆变对于编程来说是不直观的,理解逆变可能需要一定的时间。譬如在第 2 节中的 Action 例子中,你可能会想:凭什么说 Action<Animal> 是 Action<Cat> 的子类呢,Action<Object> 表示对一个 Object 进行处理,那么对一个更具体的 Cat 类的处理明显要精细于对 Animal 类的处理,岂不表明 Action<Animal> 无法胜任 Action<Cat> 的精细场景么?

    然而如果你足够了解 OOP 的继承,你应该要记住,在继承中:子类对父类已有行为的继承改造,应该是相同功能的不同实现,而不是修改覆盖父类功能,更不能是其父类功能的局部实现。 Action<> 作为一种委托(C# 委托),其 invoke 方法囊括其全部特性,那么倘若有一个 Action<Animal> 能成为一个 Action<Cat> 的子类,那么这个 Action<Animal> 的处理行为,应该是其父类 Action<Cat> 的相同功能的不同实现,否则不应该成为其子类。Action<> 作为一种库模板,没有指定行为的细节,但编程中为一个 Action<> 指定细节时,它的类型信息就不再是泛化的 Action<>,不能随便指定两个 Action<> 实例,就去讨论其类型关系。理解这些,才能更好地理解继承和变型。

  • 变型关系在编程中并不是绝对的,脱离应用场景去讨论变型关系,会破坏编程的灵活性。比如在第 2 节的例子中,列表是不变的,而在第 4 节的 sort 函数中,数组实际上被当成协变的(因为我们说了“Object数组足够泛化”这种话,说明 Object 数组可以当其他类型数组的父类),以及在各大支持泛型的语言的 sort 函数中,又可以将数组当成不变的,这是由排序行为的特殊性决定的,具体为什么,可以自己思考一下。

6. 特例(Dart 语言)

关于第 2 节提到的类型构造符原则,在 Dart 语言中却有一个特例。

Some (rarely used) coding patterns rely on tightening a type by overriding a parameter’s type with a subtype, which is invalid. In this case, you can use the covariant keyword to tell the analyzer that you are doing this intentionally. This removes the static error and instead checks for an invalid argument type at runtime.
一些(少数情况下)编程模式要求使用子类类型覆写参数类型来收紧类型本身(指拥有这个参数的类型),这在语法上不合法。这种情况下,你可以使用 covariant 关键字告诉分析器你的意图。此时静态错误被抑制,而在运行时检查参数类型的合法性。

正常来说,一个类型构造器的子类,应该接收更泛化的参数(逆变),而返回更特化的结果(协变),才能使得子类适用于所有父类场景。而 Dart 中的 covariant 关键字打破了这个约定,covariant 允许子类收紧其参数的类型(使构造器对于参数是协变的),以使子类专门化。这实际上是违背里氏替换原则的。

考虑其给出的例子:

class Animal {
  void chase(Animal x) { ... }
}

class Mouse extends Animal { ... }

class Cat extends Animal {
  @override
  void chase(covariant Mouse x) { ... }
}

给每一种动物定义了追逐其他动物的方法,而在猫的实现中,收紧了追逐对象,表明猫只能追逐老鼠。这使得 Cat 不是在任何场景下都能将类型擦除成 Animal 使用的,违背了里氏替换原则。

这种情况通常要求调用方充分了解类型构造器实例的真实类型,并且对于父类构造器类型,实际上是不建议再使用的。

7. 写在最后

标题或许更应该写成“协变的 (Covariant)、逆变的 (Contravariant)与不变的 (invariant)”,这三个形容词在描述具有这种变型特性的类型构造器,描述主体是构造器,而不指特性本身。但中文总是这样,表意高于表形,简洁重于准确。况且我们在说这三个词时,实际上更关注特性本身,所以,只要不造成误解就行。

8. 参考

  1. 协变与逆变 - 维基百科,自由的百科全书 (wikipedia.org)
  2. The covariant keyword | Dart

这篇好文章是转载于:学新通技术网

  • 版权申明: 本站部分内容来自互联网,仅供学习及演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,请提供相关证据及您的身份证明,我们将在收到邮件后48小时内删除。
  • 本站站名: 学新通技术网
  • 本文地址: /boutique/detail/tanhfifich
系列文章
更多 icon
同类精品
更多 icon
继续加载