| GZU521.COM学习网 |
|
传统的基于类的面向对象语言的一个主要特点就是inheritance, subclassing和subtyping之间的密不可分的联系。很多的面向对象语言的语法,概念,就是从这三者而来的。比如说,通过subclassing, 你可以继承父类的一些方法,而同时你又可以在子类中改写父类的方法。这个改写过的方法,通过subtyping, subsumption, 又可以从一个类型是父类的对象去调用。 但是,inheritance, subclassing, subtyping这三者并不是永远和睦相处的。在一些场合,这三者之间的纠缠不清会妨碍到通过继承或泛型得到的代码重用。因此,人们开始注意到把这三者分离开来的可能性。区分subclassing和subtyping已经很常见了。而其它的一些方法还处于研究的阶段。这一章我们将介绍这样一些方法。 8{u*UJ,q9fnY$_1 [ 本 资 料 来 源 于 贵 州 学 习 网 IT认证计算机软件水平考试 http://Www.gzU521.com ] 8{u*UJ,q9fnY$_1 一,对象类型 在早期的面向对象语言中(如simula), 类型的定义是和方法的实现是混合在一起的。这种方式违反了我们今天已经被广泛认识到的把实现和规范(specification) 分离的原则。这种分离得原则在开发是团队进行的时候尤其显得重要。 更近期一些的语言,通过引入不依赖于实现的对象类型来区分实现和规范。modula-3以及其它如Java等的支持class和interface的语言都是采用的这种技术。 在本书中,我们开始引入instancetypeof(cell)时,它代表的概念相当有限。看上去,它似乎只表示用new cell生成的对象的类型,于是,我们并不能用它来表示从其它类new出来的对象。但后来,当我们引入了subclassing, method overriding, subsumption和dynamic dispatch之后,事情变得不那么简单了。我们的instancetypeof(cell)已经可以用来表示从cell的子类new出来的对象,这些对象可以包括不是cell类定义的属性和方法。 如此看来,让instancetypeof(cell)依赖于一个具体的类似乎是不合理的。实际上,一个instancetypeof(cell)类型的对象不一定会跟class cell扯上任何关系。 它和cell类的唯一共同之处只是它具有了所有cell类定义的方法的签名(signature). 基于这种考虑,我们可以引入对象类型的语法: 针对cell类和recell类的定义: class cell is var contents: integer :=0; method get(): integer is return self.contents; end; method set(n:integer) is self.contents := n; end; end; subclass recell of cell is var backup: integer := 0;' D]J^|AUHV eG:k[ 此文转贴于我的学习网IT认证计算机软件水平考试 http://www.Gzu521.com]' D]J^|AUHV eG:k override set(n: integer) is self.backup := self.contents; super.set(n); end; method restore() is self.contents := self.backup; end; end; 我们可以给出这样的对象类型定义: objecttype cell is var contents: integer; method get(): integer; method set(n:integer); end; objecttype recell is var contents: integer; var backup: integer; method get(): integer method set(n: integer); method restore(); end; 这两个类型的定义包括了所有cell类和recell类定义的属性和方法的类型,但却并不包括实现。这样,它们就可以被当作与实现细节无关的的接口以实现规范和实现的分离。两个完全无关的类c和c’, 可以具有相同的类型cell, 而cell类型的使用者不必关心它使用的是c类还是c’类。 注意,我们还可以加入额外的类似继承的语法来避免在recell里重写cell里的方法签名。但那只是小节罢了。 二,分离subclassing和subtyping. 在我们上一章的讨论中,subtype的关系是建立在subclass关系的基础上的。但如果我们想要让type独立于class, 那么我们也需要定义独立于subclass的subtype. 在定义subtype时,我们又面临着几种选择:subtype是由类型的组成结构决定的呢?还是由名字决定呢? 由类型的组成结构决定的subtype是这样的:如果类型一具有了类型二的所有需要具备的属性和方法,我们就说类型一是类型二的subtype. 由类型名字决定的subtype是这样的:只有当类型一具有了类型二的所有需要具备的属性和方法, 并且类型一被明确声明为类型二的subtype时,我们才认可这种关系。 而如果我们的选择是一,那么那些属性和方法是subtype所必需具备的呢?哪些是可有可无的呢?5T7(BOUWJXsmR9Fme[本_文_来_源_于_我_的_学_习_网IT认证计算机软件水平考试 http://Www.GZU521.Com ]5T7(BOUWJXsmR9Fme 由组成结构决定的subtype能够在分布式环境和object persistence系统下进行类型匹配(译者注:对这点,我也不甚明了。看来,纸造得还是不够)。缺点是,如果两个类型碰巧具有了相同的结构,但实际上却风马牛不相及,那就会造成错误。不过,这种错误是可以用一些技术来避免的。 相比之下,基于名字的subtype不容易精确定义,而且也不支持基于结构的subtype. (译者按,这里,我无论如何和作者找不到同感。基于结构的subtype的缺点是一目了然,不过完美的避免的方法我却看不出来。而基于名字的subtype为什么就不能精确定义呢?c++/java/C#, 所有流行的oo语言都只支持基于名字的subtype, 也没有发现有什么不够灵活的地方。需要在不同名字但类似结构的类型之间架桥的话,adapter完全可以胜任嘛!) 目前,我们可以先定义一个简单的基于结构的subtype关系: 对两个类型o和o’, o’ <: o 当 o’ 具有所有o类型的成员。o’可以有多于o的成员。 |
责任编辑:gzu521