Scala之“逆变”合理性的思考_scala substitute-程序员宅基地

技术标签: 合理性  逆变  scala  Scala语言  Contravari  示例  

Scala之“逆变”合理性的思考

对于逆变的概念可以参考本系列的前一篇文章: Scala之类型参数化:Type Parameterization 本文的重点是要解释“逆变”的合理性。本文原文出处: http://blog.csdn.net/bluishglc/article/details/52585991 严禁任何形式的转载,否则将委托CSDN官方维护权益!

在思考“逆变”的合理性这个问题上,我们需要清晰地认识到一个前提,即父类与子类之间的关系实质,我们说:如果类A是类B的父类,那么所有出现类A声明的地方,我们都可以使用类B的实例进行替换,或者说所有适用于类A的操作同样适用于类B,简言之就是子类型可以透明无害地替换父类型(也就是里氏替换原则),因为子类型一定也是父类型,但父类型未必一定是子类型(有其他子类型),上述原则就是大家所熟知的里氏替换原则。

Liskov Substitution Principle (里氏替换原则)

It is safe to assume that a type T is a subtype of a type U if you can substitute a value of type T wherever a value of type U is required. 

The principle holds if T supports the same operations as U and all of T’s operations require less and provide more than the corresponding operations in U.

在回顾完上述表述之后,我们来重新审视一下“逆变”存在的合理性。首先定义如下一个Animal类族:

scala> class Animal
defined class Animal

scala> class Bird extends Animal 
defined class Bird

scala> class Dog extends Animal
defined class Dog

现在有f1,f2两个函数:

def f1(x: Bird): Unit // instance of Function1[Bird, Unit]
def f2(x: Animal): Unit // instance of Function1[Animal, Unit]

在这里,f1是f2的父类。为什么?我们知道,Function1的类型声明是Function1[-T1,+R],即函数是虽参数类型逆变,返回值类型协变的。其中随返回值类型协变是很容易理解的,随参数类型逆变往往让人费解,对此,我们同样使用前面提到的原则进行判定:父类可以被子类替换,反之则不可以,但是这里的情况会稍微有些复杂,因为我们要判断的是函数类型之间的可替换关系(即父子关系),我们可以认为函数是一种“复合”类型,它们的类型是由它们的参数和返回值的类型决定的,因此我们可以很自然的延展出这样一个规则:对于具有相同参数列表类型和返回值类型的函数,如果传给函数1的参数类型同样可以传给函数2,而传给函数2的参数未必都能传给函数1,也就是说,只从参数部分考量,函数1可以被函数2替换,即函数1是父类,函数2 是子类。

对于f2,我们说传给它一个Animal实例它可以工作,传给它一个Bird实例它仍然可以工作,在传给它一个Bird实例时,我们就要注意到,这时的f2(仅看参数部分)实例的类型实际上就已经变成f1了,这时所有声明使用f1类型的地方都可以用f2的实例去替换,但是反过来,所有声明了使用f2类型的地方我们是不能用f1的实例去替换的,因为对于f2来说,它可以接受Animal类型的任何其他子类型,比如Dog,但是Dog类型显然不适用于f1的。所以总结起来,f1可以被f2替换,但是f2不能被f1替换,所以f1是f2的父类型!

让我们再延伸地思考一下,我们可以说:因为f2是“消费”(consume)一个较为“通用”的父类型,这使得函数f2本身自然地能接纳和处理给定参数类型的所有子类型,也就意味着f2可以去替换或赋值给那些所有声明使用“具体”子类型为参数的函数,比如f1, 所以f1是父类,f2是子类!这种“消费”关系决定了逆变存在的理由,可以表述为PECS原理:

PECS stands for producer-extends, consumer-super.
In other words, if a parameterized type represents a T producer, use <? extends T>;
if it represents a T consumer, use <? super T>.

上述PECS原则换一种方法表述为:

G[+A]类似一个生产者,提供数据。(大部分情况下称G为容器类型)
G[-A] 是一个消费者,主要用来消费数据。(参考垃圾桶和垃圾的例子)

尽管我们依然在使用里氏替换原则来分析和识别“逆变”的场景,但是我们不能不承认这种解释依然只是一种逻辑上的逆推,它的解释总是让人觉得不是那么“解痒”,在本文的最后,我试图从正面给出一种“逆变”合理性的解释:

**我们说在现实世界里,如果有一类物品专门针对另一类物品而存在,除了人们一般认为的伴随着被处理物品的细化,处理品本身需要不断地跟进细化,这是“协变”的场景,也确实有可能会存在另外一种完全相反的情形:即伴随着被处理物品的细化,在掌握了越来越多处被理物品的信息和特征的趋势下,处理物品本身却可以变的愈发的简单(处理面变窄),反倒是那些处理更通用物品的处理类复杂的多,因为它们要考虑的可能的情况更多更复杂,那么这种情形就是典型的“逆变”!

一个典型是例子是空调和遥控器,如果说遥控器是基于空调类型的范型类,那么它天然应该是逆变的,即:RemoteController[-T], 空调品牌和型号越细化,遥控器实际上越单一,实现起来也越简单,反倒是随着空调类型不断地向上抽象,遥控器会变得越加复杂,直到面向所有空调通用的遥控器RemoteController[AirConditioner]诞生,这也就是我们见到过的那种万能遥控器。万能遥控器可以替换任何品牌和型号的遥控器,因此它是它们的子类!

我们可以看到大多数的逆变类有如下一些特点:

  • 如果逆变类有一个类族,那么这个类族不会是自上而下的树状结构,而是多条单线继承的路径组合,比如:RemoteController[AirConditioner] <: RemoteController[Haier]; RemoteController[AirConditioner] <: RemoteController[Gree]等等
  • 逆变类的父类和子类虽然作为父类和子类有替换关系,但是却没有任何继承关系,逆变类的子类之所以能够替换父类往往是它涵盖了父类的功能(针对某个更具体形变类型实现功能),而不存在继承父类实现的动作,因此逆变类的子类实现起来反而更加复杂。
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/bluishglc/article/details/52585991

智能推荐

前端开发之vue-grid-layout的使用和实例-程序员宅基地

文章浏览阅读1.1w次,点赞7次,收藏34次。vue-grid-layout的使用、实例、遇到的问题和解决方案_vue-grid-layout

Power Apps-上传附件控件_powerapps点击按钮上传附件-程序员宅基地

文章浏览阅读218次。然后连接一个数据源,就会在下面自动产生一个添加附件的组件。把这个控件复制粘贴到页面里,就可以单独使用来上传了。插入一个“编辑”窗体。_powerapps点击按钮上传附件

C++ 面向对象(Object-Oriented)的特征 & 构造函数& 析构函数_"object(cnofd[\"ofdrender\"])十条"-程序员宅基地

文章浏览阅读264次。(1) Abstraction (抽象)(2) Polymorphism (多态)(3) Inheritance (继承)(4) Encapsulation (封装)_"object(cnofd[\"ofdrender\"])十条"

修改node_modules源码,并保存,使用patch-package打补丁,git提交代码后,所有人可以用到修改后的_修改 node_modules-程序员宅基地

文章浏览阅读133次。删除node_modules,重新npm install看是否成功。在 package.json 文件中的 scripts 中加入。修改你的第三方库的bug等。然后目录会多出一个目录文件。_修改 node_modules

【】kali--password:su的 Authentication failure问题,&sudo passwd root输入密码时Sorry, try again._password: su: authentication failure-程序员宅基地

文章浏览阅读883次。【代码】【】kali--password:su的 Authentication failure问题,&sudo passwd root输入密码时Sorry, try again._password: su: authentication failure

整理5个优秀的微信小程序开源项目_微信小程序开源模板-程序员宅基地

文章浏览阅读1w次,点赞13次,收藏97次。整理5个优秀的微信小程序开源项目。收集了微信小程序开发过程中会使用到的资料、问题以及第三方组件库。_微信小程序开源模板

随便推点

Centos7最简搭建NFS服务器_centos7 搭建nfs server-程序员宅基地

文章浏览阅读128次。Centos7最简搭建NFS服务器_centos7 搭建nfs server

Springboot整合Mybatis-Plus使用总结(mybatis 坑补充)_mybaitis-plus ruledataobjectattributemapper' and '-程序员宅基地

文章浏览阅读1.2k次,点赞2次,收藏3次。前言mybatis在持久层框架中还是比较火的,一般项目都是基于ssm。虽然mybatis可以直接在xml中通过SQL语句操作数据库,很是灵活。但正其操作都要通过SQL语句进行,就必须写大量的xml文件,很是麻烦。mybatis-plus就很好的解决了这个问题。..._mybaitis-plus ruledataobjectattributemapper' and 'com.picc.rule.management.d

EECE 1080C / Programming for ECESummer 2022 Laboratory 4: Global Functions Practice_eece1080c-程序员宅基地

文章浏览阅读325次。EECE 1080C / Programming for ECESummer 2022Laboratory 4: Global Functions PracticePlagiarism will not be tolerated:Topics covered:function creation and call statements (emphasis on global functions)Objective:To practice program development b_eece1080c

洛谷p4777 【模板】扩展中国剩余定理-程序员宅基地

文章浏览阅读53次。被同机房早就1年前就学过的东西我现在才学,wtcl。设要求的数为\(x\)。设当前处理到第\(k\)个同余式,设\(M = LCM ^ {k - 1} _ {i - 1}\) ,前\(k - 1\)个的通解就是\(x + i * M\)。那么其实第\(k\)个来说,其实就是求一个\(y\)使得\(x + y * M ≡ a_k(mod b_k)\)转化一下就是\(y * M ...

android 退出应用没有走ondestory方法,[Android基础论]为何Activity退出之后,系统没有调用onDestroy方法?...-程序员宅基地

文章浏览阅读1.3k次。首先,问题是如何出现的?晚上复查代码,发现一个activity没有调用自己的ondestroy方法我表示非常的费解,于是我检查了下代码。发现再finish代码之后接了如下代码finish();System.exit(0);//这就是罪魁祸首为什么这样写会出现问题System.exit(0);////看一下函数的原型public static void exit (int code)//Added ..._android 手动杀死app,activity不执行ondestroy

SylixOS快问快答_select函数 导致堆栈溢出 sylixos-程序员宅基地

文章浏览阅读894次。Q: SylixOS 版权是什么形式, 是否分为<开发版税>和<运行时版税>.A: SylixOS 是开源并免费的操作系统, 支持 BSD/GPL 协议(GPL 版本暂未确定). 没有任何的运行时版税. 您可以用她来做任何 您喜欢做的项目. 也可以修改 SylixOS 的源代码, 不需要支付任何费用. 当然笔者希望您可以将使用 SylixOS 开发的项目 (不需要开源)或对 SylixOS 源码的修改及时告知笔者.需要指出: SylixOS 本身仅是笔者用来提升自己水平而开发的_select函数 导致堆栈溢出 sylixos

推荐文章

热门文章

相关标签