---------------------------------------------------------------------------------------------------------------------------------------------------------------
每次都对你说,我比你想象的在乎你,经过这么久,我就想说,你比我想象的爱我。
不负时光,不负你!
---------------------------------------------------------------------------------------------------------------------------------------------------------------
虚基类作用
在一个类中间接保留共同基类的多份同名成员,大部分是不需要的,因为保留多个拷贝,占用存储空间,而且增加了访问的困难,容易出错。
假设类D是类B和类C公有派生,类B和类C又是类A的派生类。如下图所示:
C++提供虚基类的方法,使得在继承时候共同基类只保留一份成员。现在把A声明为虚基类:
class A
{.......}
class B : virtual public A
{......}
class C:virtual public A
{......}
虚基类不是在声明基类时候声明的,而是在声明派生类时候声明的,指定继承方式声明的,格式如下:
class 派生类名:virtual 继承方式 基类名
在声明派生时候,将关键字virtual 加到相应的继承方式前面,当基类通过多条派生路径被一个派生类继承时,该派生只继承一次,也就是基类成员只保留一次。
虚基类的初始化
class A
{
A(int i){}
......
};
class B : virtual public A
{
B(int n):A(n){}
......
};
class C:virtual public A
{
C(int n):A(n){}
......
};
class D:public B,public C
{
D(int n):A(n),B(n),C(n){}
......
};
定义D的构造函数和以前的不同,以前只需要对直接基类进行初始化,再由直接基类对间接基类的初始化,现在,由于虚基类只有一份数据成员,所以这份数据成员必须由派生类给出。所以规定:在最后的派生类中不仅要负责直接基类进行初始化,还要负责对虚基类的初始化。
虚基类的简单应用
#include <iostream>
using namespace std;
class Person //声明公共基类Person
{
public:
Person(string na, char se, int i ):name(na),sex(se),age(i){}
protected:
string name;
char sex;
int age;
};
class Teacher:virtual public Person
{
public:
Teacher(string na, char se, int i,string tit):Person(na, se, i),title(tit){}
protected:
string title;
};
class Student:virtual public Person
{
public:
Student(string na, char se, int i,float sco):Person(na, se, i),score(sco){}
protected:
float score;
};
class Graduate:public Teacher,public Student
{
public:
Graduate(string na, char se, int i,string t,float sco,float w)
:Person(na, se, i),Teacher(na, se, i,t),Student(na, se, i,sco),wage(w){}
void show()
{
cout<<"name:"<<name<<endl;
cout<<"sex:"<<sex<<endl;
cout<<"age:"<<age<<endl;
cout<<"score:"<<score<<endl;
cout<<"title:"<<title<<endl;
cout<<"wages:"<<wage<<endl;
}
private:
float wage;
};
int main()
{
Graduate yq("yqq",'B',27,"Gradua",100,10000);
yq.show();
return 0;
}