虚基类
在说明其作用前先看一段代码
class A
{
public:
int iValue;
};
class B:public A
{
public:
void bPrintf(){cout<<“This is class B”<<endl;};
};
class C:public A
{
public:
void cPrintf(){cout<<“This is class C”<<endl;};
};
class D:public B,public C
{
public:
void dPrintf(){cout<<“This is class D”<<endl;};
};
void main()
{
D d;
cout<<d.iValue<<endl; //错误,不明确的访问
cout<<d.A::iValue<<endl; //正确
cout<<d.B::iValue<<endl; //正确
cout<<d.C::iValue<<endl; //正确
}
从代码中可以看出类B C都继承了类A的iValue成员,因此类B C都有一个成员变量iValue ,而类D又继承了B C,这样类D就有一个重名的成员 iValue(一个是从类B中继承过来的,一个是从类C中继承过来的).在主函数中调用d.iValue 因为类D有一个重名的成员iValue编译器不知道调用 从谁继承过来的iValue所以就产生的二义性的问题.正确的做法应该是加上作用域限定符 d.B::iValue 表示调用从B类继承过来的iValue。不过类D的实例中就有多个iValue的实例,就会占用内存空间。所以C++中就引用了虚基类的概念,来解决这个问题。
二义性:在多继承的场景里,当父类中存在同名变量时,子类访问父类的同名变量,将出现二义性,因为编译器不知道你将要访问的是哪个父类中的变量
class A
{
public:
int iValue;
};
class B:virtual public A
{
public:
void bPrintf(){cout<<“This is class B”<<endl;};
};
class C:virtual public A
{
public:
void cPrintf(){cout<<“This is class C”<<endl;};
};
class D:public B,public C
{
public:
void dPrintf(){cout<<“This is class D”<<endl;};
};
void main()
{
D d;
cout<<d.iValue<<endl; //正确
}
在继承的类的前面加上virtual关键字表示被继承的类是一个虚基类,它的被继承成员在派生类中只保留一个实例。例如iValue这个成员,从类 D这个角度上来看,它是从类B与类C继承过来的,而类B C又是从类A继承过来的,但它们只保留一个副本。因此在主函数中调用d.iValue时就不 会产生错误。