王朝网络
分享
 
 
 

面向对象设计(OOD)中的替换原则

王朝other·作者佚名  2006-12-25
宽屏版  字体: |||超大  

我们知道,在面向对象语言中,公有继承是IS-A的关系,也就是说子类是一种基类,就像说轿车是一种汽车一样。但是,有时候逻辑上正确的公有继承却会违反替换原则。面向对象设计中的替换原则是:

子类必须能够替换掉它们的基类。

也就是说,代码中基类出现的地方都能用子类来替换,就跟汽车能用的地方都能用轿车一样。但是,如果设计不合理,就会违反这个原则,给开发带来隐患。

下面就以一个C++的例子来说明:

class CShape

{

public:

virtual ~CShape() {};

virtual double GetArea() = 0;

};

class CRectangle : public CShape

{

public:

virtual ~CRectangle() {};

double GetArea() { return m_dWidth * m_dHeight; };

void SetWidth(double dWidth) { m_dWidth = dWidth; };

void SetHeight(double dHeight) { m_dHeight = dHeight; };

double GetWidth() const { return m_dWidth; };

double GetHeight() const { return m_dHeight; };

private:

double m_dWidth;

double m_dHeight;

};

现在,我们需要一个正方形类CSquare。我们知道,正方形是一种宽和高相同的长方形,也就是说正方形和长方形符合IS-A的关系。那我们需要的类CSquare是否应该从类CRectangle来继承呢?

我们现在先来看看类CSquare从类CRectangle继承的情况:

class CSquare : public CRectangle

{

public:

virtual ~CSquare() {};

void SetWidth(double dWidth)

{

CRectangle::SetWidth(dWidth);

CRectangle::SetHeight(dWidth);

};

void SetHeight(double dHeight)

{

CRectangle::SetHeight(dHeight);

CRectangle::SetWidth(dHeight);

};

};

类CSquare中对方法SetWidth和方法SetHeight的重载解决了正方形的宽和高相等的问题。但是请考虑下面的代码:

void FncTest(CRectangle& r)

{

r.SetWidth(13);

assert(r.GetArea() == 169);

}

void Test()

{

CSquare s;

s.SetWidth(5);

FncTest(s);

}

当我们向函数FncTest传递一个指向类CSquare对象的引用的时候,这个函数就会出现断言错误,因为类CSquare对象的高度没有改变。这就违反了替换原则,函数FncTest中的类型为基类CRectangle的参数不能用子类CSquare来替换。错误原因很简单,类CRectangle中的方法SetWidth和方法SetHeight不是虚函数,因此不具备多态性。在这种情况下,我们就必须去修改类CRectangle。下面是修改后的代码:

class CRectangle : public CShape

{

public:

virtual ~CRectangle() {};

double GetArea() { return m_dWidth * m_dHeight; };

virtual void SetWidth(double dWidth) { m_dWidth = dWidth; };

virtual void SetHeight(double dHeight) { m_dHeight = dHeight; };

double GetWidth() const { return m_dWidth; };

double GetHeight() const { return m_dHeight; };

private:

double m_dWidth;

double m_dHeight;

};

class CSquare : public CRectangle

{

public:

virtual ~CSquare() {};

virtual void SetWidth(double dWidth)

{

CRectangle::SetWidth(dWidth);

CRectangle::SetHeight(dWidth);

};

virtual void SetHeight(double dHeight)

{

CRectangle::SetHeight(dHeight);

CRectangle::SetWidth(dHeight);

};

};

这样,上面提到的问题就解决了,修改也不是很复杂。然而,如果派生类的创建会导致我们修改基类,这就意味着设计是有缺陷的。我们先不管是否有设计缺陷,而是先接受上面的修改。现在类CRectangle和类CSquare都能工作,在代码中也可以接受指向类CRectangle的指针或引用的函数传递类CSquare,而类CSquare也可以很好的保持正方形的特性。看起来,这样的设计已经符合替换原则了。但是,等等,请看下面的函数:

void FncTest(CRectangle& r)

{

r.SetWidth(5);

r.SetHeight(6);

assert(r.GetArea() == 30);

}

对这个函数来说如果传递来的是类CRectangle的对象,则运行正确;如果传递来的是类CSque的对象,那就会出现断言错误!仍旧是违反了替换原则!

上面的例子说明了,即使两个类是IS-A的关系,也不一定要用公有继承来实现。

其实,在面向对象设计(OOD)中,IS-A关系是就行为方式而言的。在本例中,虽然类CRectangle和类CSquare在逻辑上是IS-A关系,但是他们的行为是不同的,因此也就不能让类CSquare从类CRectangle公有继承,而是从类CShape公有继承。

下面是正确的代码:

class CShape

{

public:

virtual ~CShape() {};

virtual double GetArea() = 0;

};

class CRectangle : public CShape

{

public:

virtual ~CRectangle() {};

double GetArea() { return m_dWidth * m_dHeight; };

void SetWidth(double dWidth) { m_dWidth = dWidth; };

void SetHeight(double dHeight) { m_dHeight = dHeight; };

double GetWidth() const { return m_dWidth; };

double GetHeight() const { return m_dHeight; };

private:

double m_dWidth;

double m_dHeight;

};

class CSquare : public CShape

{

public:

virtual ~CSquare() {};

double GetArea() { return m_dWidth * m_dWidth; };

void SetWidth(double dWidth) { m_dWidth = dWidth; };

double GetWidth() const { return m_dWidth; };

private:

double m_dWidth;

};

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
>>返回首页<<
推荐阅读
 
 
频道精选
 
静静地坐在废墟上,四周的荒凉一望无际,忽然觉得,凄凉也很美
© 2005- 王朝网络 版权所有