上章最后的例子是为下面讨论一个被称之为“菱形问题”作铺垫的,在Java中没有多重继承,也许没有这种现象,C++中很容易出现。由一个基类派生出两个类出来,以后新定义一个类,并从这两个类多重继承,这样就出现菱形问题了。也就是说,基类的公有或保护成员,必然被两个派生类同时继承,这两个类同时派生一个新类时,同名成员就产生了冲突。
下面我们先从简单问题入手,先不要看结果图,考虑一下结果应该是什么。小雅当初认为三行输出的ID都应为“WD8503025”,但事实不是这样。
#include
<iostream>#include
<string>using namespace
std;class
CBase {public
: string id; };class
CDerive1 :public
CBase {public
:void
show1() { cout << "CDerive1: " << id << endl; } };class
CDerive2 :public
CBase {public
:void
show2() { cout << "CDerive2: " << id << endl; } };class
CSon :public
CDerive2,public
CDerive1 { };int
main ( ) { CSon s; s.CDerive1::id = "WD8503026"; s.CDerive2::id = "WD8503027"; s.CBase::id = "WD8503025"; s.show1(); s.show2(); cout << "BASE: " << s.CBase::id << endl;return
0; }
通过设置断点不难看出,当前实例是CBase的“孙子”,而“父亲”有2个,每个“父亲”都将“爷爷”复制了一份。如果不指定哪个“父亲”的“父亲”,默认将第一个继承的“父亲”的“父亲”当作“爷爷”。上例中先继承CDerive2类,所以31行和34行的“s.CBase::id”等价于“s.CDerive2::CBase::id”。
下面的例子是先定义一个“孙子”的实例,并将地址输出。再将这个实例的地址分别赋给CDerive1和CDerive2类型的指针变量,并输出指针地址。再将这2个地址分别赋给CBase的2个指针变量,并输出其地址。大家仍然不看结果,考虑一下答案应该是什么?
#include
<iostream>#include
<string>using namespace
std;class
CBase { string id;public
:void
show() { cout << id << endl; } };class
CDerive1 :public
CBase { };class
CDerive2 :public
CBase { };class
CSon :public
CDerive2,public
CDerive1 { }int
main ( ) { CSon s; cout << &s << endl; cout << "---------" << endl; CDerive1 *pd1 = &s; cout << pd1 << endl; CDerive2 *pd2 = &s; cout << pd2 << endl; cout << "---------" << endl; CBase *pb1 = pd1; cout << pb1 << endl; CBase *pb2 = pd2; cout << pb2 << endl; //CBase *pb = &s; //编译有错return
0; }
从上例可以看出,由于2个“父亲”因而复制出2个“爷爷”,2个“爷爷”的地址也不同。
解决以上问题只要用C++的“虚继承”就可以了。“虚继承”就是在实例中,基类不管继承多少个,只复制一份。
#include
<iostream>#include
<string>using namespace
std;class
CBase {public
: string id; }; //因为虚继承,CBase类在此不产生副本class
CDerive1 :virtual public
CBase {public
:void
show1() { cout << "CDerive1: " << id << endl; } }; //因为虚继承,CBase类在此不产生副本class
CDerive2 :virtual public
CBase {public
:void
show2() { cout << "CDerive2: " << id << endl; } };class
CSon :public
CDerive2,public
CDerive1 { };int
main ( ) { CSon s; s.CDerive1::id = "WD8503026"; s.CDerive2::id = "WD8503027"; s.CBase::id = "WD8503025"; s.show1(); s.show2(); cout << "BASE: " << s.CBase::id << endl;return
0; }
因为2个“父亲”都不产生“爷爷”的副本,所以“孙子”这儿就只有一个“爷爷”的副本。