C++:多重继承
MI描述的是有多个直接基类的类。与单继承一样,公有MI表示的也是is-a关系。例如,Worker派生出Singer和Waiter,再从Singer和Waiter派生出SingingWaiter。但多重继承会带来许多问题,最主要的两个便是
- 从两个不同的基类继承的同名方法
- 从两个或更多相关基类哪里继承同一个类的多个实例
就上述两个问题,下面根据举例Worker继承给出相关解决方法
有多少Worker
Worker派生出Singer和Waiter,Singer和Waiter再派生出SingerWaiter,则SingingWaiter将包含两个Worker,这将可能出现二义性
SingingWaiter ed;
Worker*pw=&ed;//是哪个Worker的地址呢?
所以这里需要使用类型转换来指定对象。
Worker *pw1=(Waiter*) &ed;//Waiter中的Worker地址
Worker *pw2=(Singer*) &ed;//Singer中的Worker地址
这使得使用基类指针来引用不同对象变得复杂,但真正的问题是为什么需要多个Worker?SingingWaiter的对象是一个单独的人,不该带有两个Worker对象。为此我们需要引入虚基类来确保从多个类派生出的对象只继承一个基类对象。
虚基类
class Singer:virtual public Worker{...}
class Waiter:public virtual Worker{...}//virtual的位置无关紧要
class SingingWaiter:public Singer,public Waiter{...}
上述代码声明使得Worker被用作Singer和Waiter的虚基类,且SingingWaiter对象只包含了一个Worker子对象。
使用虚基类时,需对类构造函数采用一种新的方法。C++在基类是虚的时,禁止信息通过中间类自动传递给虚类。因此派生类实现的多重继承需要直接调用基类的构造函数(与非虚基类不允许越过上一级不同)。
SingingWaiter(const Worker&wk,int p=0,int v=Singer::other):Waiter(wk,p),Singer(wk,v){}
//这里wk信息不会传递到虚基类,因为两条途径不同会产生冲突
//会自动调用Worker的默认构造函数SingingWaiter(const Worker&wk,int p=0,int v=Singer::other):Worker(wk),Waiter(wk,p),Singer(wk,p){}
//这样显式调用构造函数Worker可阻止Worker默认构造函数的调用
注意:这种用法在虚基类是合法的,且必须这样做;但对于非虚基类,这样是非法的。
使用哪个方法
我们希望通过SingingWaiter调用show()方法(同时被Singer和Waiter拥有),但直接调用会导致二义性,不知道会调用哪个直接祖先的方法。于是,我们可以通过使用作用域解析运算符来表示意图。
SingingWaiter n("ew",2005,6,soprano)
n.Singer::Show();//调用Singer的版本
但更好的方法是在SingingWaiter中重新定义Show(),并指出希望使用哪一个Show()。
void SingingWaiter::Show()
{Singer::Show();
}//希望SingingWaiter使用Singer版本的Show()
在单继承中可以使用递增的方式显示类的信息(基类显示基类的,派生类调用基类方法加上派生类的信息,以此类推)。但多重继承不行,这样会忽略部分信息或是重复显示信息。所以多重继承应使用模块化方法:提供一个Worker组件的方法和一个只显示Waiter组件或Singer组件的方法,再在SingingWaiter方法中调用上述方法。
另一种方法是将所有数据组件都设置为保护的而非私有的。
虚基类与支配
使用虚基类将改变C++解析二义性的方式。使用虚基类时,派生类使用从多个基类那里继承得到的同名成员名,如果不使用类名限定,也不一定会导致二义性。这种情况下,如果某个名称优先于其他所有名称,则使用它且不导致二义性。
如何判断优先性呢?派生类中的名称优先于直接或间接祖先中的相同名称。
class B
{
public:short q();...
};class C:virtual public B
{
public:short q;int omg()...
};
class D:public C
{
...
};
class E:virtual public B
{
private:int omg();...
};class F:public D,public E
{...
};
类C中q()定义优先于类B中的q()定义,因为C时从B类派生来的。任何一个omg()定义都不优先于其他omg()定义,因为C和E都不是对方的基类。(虚二义性与访问规则无关)