如果一个派生类从多个基类派生,而这些基类又有一个共同的基类,则在对该基类中声明的名字进行访问时,可能产生二义性
class A{public: int a;};
class A1:public A{public: int a1;};
class A2:public A{public: int a2;};
class B:public A1,public A2{public: int b;};
/main
{
B b;
b.A::a; //erroe 不知道从哪里继承
b.A1::a; //ok
b.A2::a; //ok
}
建立B类的对象时,A的构造函数被调用了两次;一次是由A1调用,一次是由A2调用,所以初始化B的对象时包含了两个A类的子对象。
总结:
1、如果一个派生类从多个基类派生,而这些基类又有一个共同的基类,则在对该基类中声明的名字进行访问时,可能产生二义性
2、 如果在多条继承路径上有一个公共的基类,那么在继承路径的某处汇合点,这个公共基类就会在派生类的对象中产生多个基类子对象
3、 要使这个公共基类在派生类中只产生一个子对象,必须对这个基类声明为虚继承,使这个基类成为虚基类。
4、 虚继承声明使用关键字 virtual
class A{public: int a;};
class A1: virtual public A{public: int a1;};
class A2: virtual public A{public: int a2;};
class B:public A1,public A2{public: int b;};
/main
{
B b1;
b1.a; //ok
b1.A1::a; //ok
b1.A2::a; //ok
}
B的对象中只存在一个A类,因此通过B类对象直接访问a,或者B.A1::a&B.A2::a,所访问的都是那个唯一的基类子对象,引用时同一个基类A的子对象。
构造函数的调用次数:
//A类构造函数
//A1类构造函数
//A2类构造函数
//B类构造函数