多态性如何提高程序的可扩展性?

[复制链接]
查看11 | 回复0 | 2007-7-4 12:33:04 | 显示全部楼层 |阅读模式
多态多态(polymorphism)在一些编程教材中被弄得很神秘,而在另外一些教程中则被忽略,其实它不过是C++语言所支持的一个简单而有用的概念。按照C++标准所言,“多态类型”旧时代由虚函数的类类型(class type)。从设计的角度来看,“多态对象”就是一个具有不止一种类型的对象,而“多态基类”则是一个为满足多态对象的使用需求而设计的基类。 让我们看一个金融期权的类型AmOption,如图1所示。 +-------------------------------------------------------+ |
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对象应该和Option对象的行为完全一致(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的多态代码也能操作它,这全托"对具体类型不知情"的福,如果以后这个对象消失了,也犯不着去挂念它。 出于同样的原因,像AmOption和EurOption这样具体的期权类型只需要知道积累就可以了,改变通用代码对他们毫无影响。原则上,基类可以不知道出自身以外的任何事物。从实践的角度来看,对其接口的设计要考虑与其用户的需求,并且应该以这样的方式进行设计:派生类很容易的推知并实现其合同。然而,基类应该对其派生类的具体细节全然不知,因为知道这些会不可避免的致使类层次结构上添加或删除派生类变得困难。 和生活中一样,在面向对象设计中,“不知情”或“忽略”也是天赐之福。
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

主题

0

回帖

4882万

积分

论坛元老

Rank: 8Rank: 8

积分
48824836
热门排行