当在多条继承路径上有一个公共的基类,在这些路径的某几条汇合处,这个公共的基类就会产生多个实例(或多个副本),若只想保存这个基类的一个实例,可以将这个公共基类说明为虚基类、
class CBase { };
class ChildA1:virtual public CBase{ };
class ChildA2:virtual public CBase{ };
class ChildB:public ChildA1,ChildA2{ };
则在类ChildB的对象中,仅有类CBase的一个对象数据
虚基类的特点:
(1):虚基类构造函数的参数必须由最新派生出来的类负责初始化(即使不是直接继承).
(2)虚基类的构造函数先于非虚基类的构造函数执行。
下面看一段程序的输出结果:
1 #include "stdafx.h" 2 #include<iostream> 3 using namespace std; 4 5 class CBase{ 6 protected: 7 int a; 8 public: 9 CBase(int na) 10 { 11 a = na; 12 cout << "CBase constructor!" << endl; 13 } 14 ~CBase() 15 { 16 cout << "CBase deconstructor!" << endl; 17 } 18 }; 19 20 class ChildA1: virtual public CBase 21 { 22 public: 23 ChildA1(int na):CBase(na) 24 { 25 cout << "ChildA1 constructor!" << endl; 26 } 27 ~ChildA1() 28 { 29 cout << "ChildA1 deconstructor!" << endl; 30 } 31 int GetA() 32 { 33 return a; 34 } 35 }; 36 37 class ChildA2 : virtual public CBase 38 { 39 public: 40 ChildA2(int na):CBase(na) 41 { 42 cout<< " ChildA2 constructor!" << endl; 43 } 44 ~ChildA2() 45 { 46 cout<< "ChildA2 deconstructor!" << endl; 47 } 48 int GetA() 49 { 50 return a; 51 } 52 }; 53 54 class ChildB:public ChildA1,public ChildA2 55 { 56 public: 57 ChildB(int a1,int a2,int a3):ChildA1(a1),ChildA2(a2),CBase(a3) 58 { 59 cout << "ChildB constructor!!" << endl; 60 } 61 ~ChildB() 62 { 63 cout << "ChildB deconstructor!" << endl; 64 } 65 }; 66 67 int main() 68 { 69 ChildB childb(100,200,300); 70 //得到从ChildA1继承的值 71 cout<<" from ChildA1 : a = "<<childb.ChildA1::GetA(); 72 //得到从ChildA2继承的值 73 cout<<" from ChildA2 : a = "<<childb.ChildA2::GetA()<<endl<<endl; 74 return 0; 75 }
程序输出的结果:
从上例中可以看出来,在类ChildB的构造函数初始列表中,调用了间接基类CBase的构造函数,这对于非基类是非法的,但对于虚基类则是合法而且是必要的。
从输出的结果可以看出来,其公共基类的构造函数只调用了一次,并且优先于非虚基类的构造函数调用,并且发现,子派生类的对象childb的成员变量的值只有一个,所以当公共基类CBase被声明为虚基类,虽然它成为ChildA1和ChildA2的公共基类,但子派生类ChildB中也只有它的一个备份.
原文:http://www.cnblogs.com/cxq0017/p/6484539.html