《Effective C++》 40:明智而审慎地使用多重继承
注:审慎,指周密而慎重。
多重继承(multiple inheritance; MI),单一继承(single inheritance; SI)
在多重继承下,由于子类可能继承自多个父类的相同名称(如函数名),因此会导致较多的歧义。
如下面这段代码:
#include <iostream>
using namespace std;
//名词
class Noun
{
public:
//做主语
void Subject()
{
cout << "I can be a subject directly." << endl;
}
};
//动词
class Verb
{
public:
void Turn()
{
Subject();
}
private:
//做主语
void Subject()
{
cout << "I'll add -ing." << endl;
}
};
//名词+动词
class NandV
: public Noun,
public Verb
{
};
int main()
{
//reserve既是动词也是名词
NandV reserve;
reserve.Subject();
return 0;
}
错误提示: errorC2385:对“Subject”的访问不明确。
这是由于在看到某个函数是否可取用(是否有权限)之前,C++首先要确认这个函数对此调用而言是最佳的匹配。
找到最佳的匹配函数 --->检验其可取性
因此在匹配Subject()函数的时候就会造成歧义。
为了消除这个歧义,必须指出调用的具体是哪个基类的函数。
如:
reserve.Noun::Subject();
《 Effective C++ 》所谓的 “钻 石型多重继承 ” ,如下:
IOFile会沿着每条路径复制File内的成员,而实际上,IOFile只应该有一组File内的成员。
针对这种情况,我们可以使用virtual继承来避免重复的成员复制。
如下:
#include <iostream>
using namespace std;
class File
{
//......
};
class InputFile
: virtual public File
{
//......
};
class OutputFile
:virtual public File
{
//......
};
class IOFile
: public InputFile,
public OutputFile
{
//......
};
int main()
{
IOFile file;
return 0;
}
《EffectiveC++》建议非必要情况下不使用虚继承,
必须使用虚继承的情况下,尽可能避免在virtual base class里面放置数据。
但是,virtual public继承是有空间的代价的。