在上一节中,有两个身份证号显然是不合理的。为此,可以把class Person这个共同基类设置为虚基类,这样,从不同路径继承来的同名数据成员在内存中就只有一个拷贝,同名函数也只有一种映射。
虚基类定义方式
虚基类(virtual base class)定义方式如下:
class 派生类名:virtual 访问限定符 基类类名{...};
或:
class 派生类名:访问限定符 virtual 基类类名{...};
其中:virtual 关键字只对紧随其后的基类名起作用。
例如:
//学生类定义:
class Student::virtual public Person{...};
//教职工类定义:
class Employee::virtual public Person{...};
采用虚基类的多重继承的特点
采用虚基类后,在职研究生类对象的储存如下图所示。
与8.3节中的图8.4(b)不同的是:在Person的位置上放的是指针,两个指针都指向Person成员存储的内存。这种继承称为虚拟继承(virtual inheritance)。
采用虚基类的多重继承的构造与析构的次序
在派生类对象的创建中,构造次序如下:
- 虚基类的构造函数被调用,并按它们声明的顺序构造;
- 非虚基类的构造函数按它们声明的顺序调用;
- 成员对象的构造函数;
- 派生类自己的构造函数被调用。
析构的次序与构造的次序相反。
应用举例
【例8.3】在采用虚基类的多重继承中,构造与析构的次序。
//【例8.3】在采用虚基类的多重继承中,构造与析构的次序。 #include<iostream> using namespace std; class Object{ public: Object(){cout<<"constructor Object ";} ~Object(){cout<<"deconstructor Object ";} }; class Bclass1{ public: Bclass1(){cout<<"constructor Bclass1 ";} ~Bclass1(){cout<<"deconstructor Bclass1 ";} }; class Bclass2{ public: Bclass2(){cout<<"constructor Bclass2 ";} ~Bclass2(){cout<<"deconstructor Bclass2 ";} }; class Bclass3{ public: Bclass3(){cout<<"constructor Bclass3 ";} ~Bclass3(){cout<<"deconstructor Bclass3 ";} }; class Dclass:public Bclass1,virtual Bclass3,virtual Bclass2{ Object object; public: Dclass():object(),Bclass2(),Bclass3(),Bclass1(){cout<<"派生类建立! ";} ~Dclass(){cout<<"派生类析构! ";} }; int main(){ Dclass dd; cout<<"主程序运行! "; return 0; }
运行结果
Constructor Bclass3 //第一个虚拟基类,与派生类析构函数排列无关
Constructor Bclass2 //第二个虚拟基类
Constructor Bclass1 //非虚拟基类
Constructor Object //对象成员
派生类建立!
主程序运行!
派生类析构!
deconstructor Object //析构次序相反
deconstructor Bclass1
deconstructor Bclass2
deconstructor Bclass3
示例 虚基类在多层多重继承中的应用——在职研究生类定义
//【例8.4】虚基类在多层多重继承中的应用--在职研究生类定义。 #include<iostream> #include<string> using namespace std; enum Tsex{mid,man,woman}; //为简化,本例定义学生类时课程省略,并全部用string字符串 class Person{ string IdPerson; //身份证号 string Name; //姓名 Tsex Sex; //性别 int Birthday; //生日,格式1986年8月18日写作19860818 string HomeAddress; //家庭地址 public: Person(string, string,Tsex,int, string); Person(); ~Person(); void PrintPersonInfo(); //其他接口函数 }; Person::Person(string id, string name,Tsex sex,int birthday, string homeadd){ cout<<"构造Person"<<endl; IdPerson=id; Name=name; Sex=sex; Birthday=birthday; HomeAddress=homeadd; } Person::Person(){ cout<<"构造Person"<<endl; IdPerson='