+-------------------------------------------------------+
| Deal iceable |
| ^ ^ |
| | | |
| +------------------------------+ |
| | Option | |
| +------------------------------+ |
| ^ ^ |
| | | |
| AmOption EurOption |
+------------------------------------------------------------------+图1 在一个金融期权层次结构中多态的作用。AmOption有4种类型
AmOption对象同时具有四种类型:AmOption\Option\Deal\Priceable。由于一个类
型是一组操作,因此,AmOption对象通过其4个接口中的任何一个进行操纵。这意味着一
个AmOption对象可以被针对Deal\Priceable\Option接口编写的代码所操纵,从而允许
AmOption的实现利用或复用所有那些代码。对于AmOption这样的多态类型,从基类继承
的最重要的东西就是他的接口,而不是他的实现。事实上,一个基类仅仅由接口组成不
但常见,而且通常正是我们所希望的。
当然,这里还有一个需要注意的地方。如果让这种优势能够发挥出来,一个良好设计
的多态类对于他的每个基类而言必须是可替换的。换句话说,如果针对Option接口编写的
通用代码接受的是一个AmOption对象,那么该对象的行为最好就像一个Option对象。
数)。实际上一个多态基类想象成一份合同更好理解一些。这个基类对其借口的用户作了
某些承诺,这些承诺包括郑重的语法承诺,即约定的成员函数可以通过一些特定类型的实
参进行调用,以及不容易验证的语义上的承诺,即当一个特定的成员函数被调用时将会发
生什么实际情况。像AmOption和EurOption这样的具体派生类被称为“转包者”,他们实现
Option与其客户签订的合同,如图2所示。
+--------------------------------+
+--------------+ |class Option |
|针对Option | |{ |
|接口编写的 |< --> | virtual price()=0; |
|代码 | | update(); |
+--------------+ |}; |
|---------------------------------+
|class AmOption:public Option |
|{ |
| virtual price(); |
|}; |
|----------------------------------+
|class EurOption:public Option |
|{ |
| virtual price(); |
|}; |
+----------------------------------------+图2,一个多态的承包者及其“转包者”
举了例子,如果Option具有一个纯虚成员函数Price,其作用是给出Option的当前值,
那么AmOption和EurOption都必须实现这个函数。我们显然不会为这两种类型的Option实
现完全一致的行为,但他们都应该计算并返回一个价格(price),而不应该去拨打一个电
话或者去玩游戏:)
另一方面,如果我们要去访问同一个对象的两种不同接口的price函数,那么我们应
该得到相同的结果。就本质而言,每一个调用都应该绑定到同一个函数:
AmOption *d=new AmOption;
Option *b=d;
d->price();
b->price(); //其实就是说虚函数实际上还是调用的AmOption的。
这是有意义的。假如我问你“那个美国期权的当前值是什么?”,我期望得到与下面
简短提问方式相同的答案:“那个期权的当前值是什么?”
b->update();
d->update(); //都是调用的b里面的update
正是基于提供的合同允许针对基类借口编写的“多态”代码对特定的合同起作用,同时
有助于对派生类的存在保持“健康的不知情”。换句话说,多态代码可能正在操作AmOption
和EerOption对象,但除非特别关心他们到底是什么对象,否则均被视为Option对象。各
种各样“具体的”Option类型可以被添加或删除而不会影响到只关心基类Option的通用代码。
比方说,如果某一个地方出现一个AsianOption对象,那么只知道Option的多态代码也能
操作它,这全托"对具体类型不知情"的福,如果以后这个对象消失了,也犯不着去挂念它。
以了,改变通用代码对他们毫无影响。原则上,基类可以不知道出自身以外的任何事物。
从实践的角度来看,对其接口的设计要考虑与其用户的需求,并且应该以这样的方式进
行设计:派生类很容易的推知并实现其合同。然而,基类应该对其派生类的具体细节全
然不知,因为知道这些会不可避免的致使类层次结构上添加或删除派生类变得困难。
多态我的理解是一个基类,有N个子类继承他,N个子类在实现同一个基类方法的时候,可以返回多种状态,^_^,不知道理解的对不?呵呵
覆盖(override)和重载(overload)。覆盖是指子类重新定义父类的方法(函数)的做法。而重载,是指允许存在多个同名函数,而这些函数的参数表不同(或许参数个数不同,或许参数类型不同,或许两者都不同)。