Guru of the Week:#39 多重继承 Ⅲ.

类别:VC语言 点击:0 评论:0 推荐:

 

/*此文是译者出于自娱翻译的GotW(Guru of the Week:http://www.gotw.ca/gotw/index.htm)系列文章的一篇,原文的版权是属于Hub Sutter(著名的C++专家,《Exceptional C++》的作者)。此文的翻译没有征得原作者的同意,只供学习讨论。——译者:黄森堂*/

#39 多重继承 Ⅲ.

难度:4/10

覆盖继承来的虚函数是很容易 -- 只要你不要在具有相同特征(原文:signature)的两个基类里尝试覆盖里面的虚函数,当基类来自不同的供应商的时候就可能发生!

问题:

1.思考以下两个类:

class B1 { public: virtual int ReadBuf( const char* ); // ... }; class B2 { public: virtual int ReadBuf( const char* ); // ... };

它们都是在基类中明显地使用,但它们不以其它发生关系,它们的ReadBuf函数是在做不同的事,且类是由不同的供应商提供的不同的库。

示范:如何写一个类D,该类公共派生于B1与B2(class D:public B1,public B2),并且重载ReadBufs来做不同的事情.

解决方法:

示范:如何写一个类D,该类公共派生于B1与B2(class D:public B1,public B2),并且重载ReadBufs来做不同的事情.

以下代码是天真的想法,它不能工作:

class D : public B1, public B2 { public: int ReadBuf( const char* ); // overrides both B1::ReadBuf and B2::ReadBuf };

这个覆盖两个函数,例它们具有相同的实现,然面,问题的要点是覆盖这两个函数来做不同的事,你不能仅仅在D::ReadBuff里面依赖对它的调用来"转变"行为,因为当你深入D::ReadBuf内部时,那儿绝对不会告诉你基本接口已经使用了。

更改虚函数名

如果两个继承过来的函数是具有不同的特征(原文:signature),那么这儿就没有问题:像通常一样,我们仅仅需要覆盖它们,如果出现前面所说的情形,只能通过改变函数名来做到最小的改变。

习惯的做法是更改基类函数的署名来创建从基类派生的中间类,定义新的虚函数,并覆盖继承过来的版本且调用新的函数:

class D1 : public B1 { public: virtual int ReadBufB1( const char* p ) = 0; int ReadBuf( const char* p ) // 覆盖继承过来的函数 { return ReadBufB1( p ); } // 调用新的函数 }; class D2 : public B2 { public: virtual int ReadBufB2( const char* p ) = 0; int ReadBuf( const char* p ) // 覆盖继承过来的函数 { return ReadBufB2( p ); } // 调用新的函数 };

D1与D2可能也需要复制B1与B2的构造器,那D就可以调用它们,但这儿请注意:D1与D2是抽象类,所以它们不需要复制任何的B1与B2的函数与重载符、赋值运算符等:

现在我们可以简单地写:

class D : public D1, public D2 { public: int ReadBufB1( const char* ); int ReadBufB2( const char* ); };

派生类只需要知道它们不必覆盖ReadBuf函数。

本文地址:http://com.8s8s.com/it/it3281.htm