注意:本文中代码均使用 Qt 开发编译环境
《虚基类》
当某类的部分或者全部直接基类是从另一个共同基类派生而来时,在这些直接基类中,从上级共同基类继承来的成员就拥有相同的名称。在派生类的对象中,这些同名数据成员在内存中同时拥有多个拷贝,同一函数名会有多个映射。我们可以用作用域分辨符来唯一标识并分别访问它们。
也可以将共同的基类设置为虚基类,这时从不同的路径继承过来的同名数据成员在内存中就只有一个拷贝,同一个函数名也只有一个映射。
语法:
class 派生类名:virtual 继承方式 基类名
上述语句声明基类为派生类的虚基类。在多继承情况下,虚基类关键字的作用范围和继承方式关键字相同,只对紧跟其后的基类起作用。声明了虚基类以后,虚基类的成员在进一步派生的过程中和派生类一起维护同一个内存数据拷贝。
举例:
#include <QCoreApplication>
#include <QDebug>
class B0 {
public:
int nV;
void fun(){qDebug()<<"Memberof B0 and nV = "<<nV;}
};
class B1 : virtual public B0 {
public:
int nV1;
};
class B2 : virtual public B0 {
public:
int nV2;
};
class C : public B1, public B2 {
public:
int nVc;
void fun(){qDebug()<<"Memberof C";}
};
int main(int argc, char *argv[])
{
QCoreApplication a(argc, argv);
C c1;
c1.B1::nV = 2;
c1.B1::fun();
c1.B2::nV = 3;
c1.B2::fun();
c1.B1::fun();
return a.exec();
}
运行结果:
《虚基类及其派生类构造函数》
如果虚基类声明有非默认形式的(即带形参的)构造函数,并且没有声明默认形式的构造函数,这时,在整个继承关系中,直接或者间接继承虚基类的所有派生类,都必须在构造函数的成员初始化表中列出对虚基类的初始化。
例如,示例:
#include <QCoreApplication>
#include <QDebug>
class B0 {
public:
B0(int n){nV = n;}
int nV;
void fun(){qDebug()<<"Memberof B0 and nV = "<<nV;}
};
class B1 : virtual public B0 {
public:
B1(int a):B0(a){}
int nV1;
};
class B2 : virtual public B0 {
public:
B2(int a):B0(a){}
int nV2;
};
class C : public B1, public B2 {
public:
C(int a):B0(a),B1(a),B2(a){}
int nVc;
void fun(){qDebug()<<"Memberof C";}
};
int main(int argc, char *argv[])
{
QCoreApplication a(argc, argv);
C c1(1);
c1.B1::fun();
c1.B2::fun();
c1.fun();
c1.B1::nV = 2;
c1.B1::fun();
c1.B2::nV = 3;
c1.B2::fun();
c1.B1::fun();
return a.exec();
}
运行结果: