/*此文是译者出于自娱翻译的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函数。