虚基类

class B

{

public:

                 int b;

};


class C1 :virtual public B

{

public:

                 int c1;

};

class C2 :virtual public B

{

public:

                 int c2;

};

class D :public C1, public C2

{

public:

                  int d;

};

int main()

{

                cout << sizeof(D ) << endl;     //24


                 B b;

                 D d;

     


                system( "pause");

                 return 0;

}


在类D中添加函数,改为:

class D :public C1, public C2

{

public:

                 void Display()

                {

                                c1 = 0x02;

                                 C2::b = 0x03;

                                c2 = 0x04;

                                d = 0x05;

                  cout << "this=" << this << endl << endl;

                  cout << "&C1::b=" << &(C1 ::b) << endl;

                  cout << "&c1=" << &c1 << endl;

                  cout << "&C2::b=" << &(C2 ::b) << endl;

                  cout << "&c2=" << &c2 << endl;

                   cout << "&d=" << &d << endl;


                  cout << b << endl;

                }

                 int d;

};

主函数写为:

int main()

{

                cout << sizeof(D ) << endl;  //24


                 B b;

                 D d;

                d.Display();


                 C1 c;

                system( "pause");

                 return 0;

}

调试截图为:

虚基类_第1张图片

虚基类_第2张图片

原理分析:

因为类B是类C1和类C2的基类,所以C1和C2都会继承B的数据成员b,所以C1和C2中会存在同名成员,当类D继承类C1和类C2的时候,就会有两个数据成员b,这样会保留多份数据成员的拷贝,不仅占用较多的存储空间,还增加了访问这些成员的困难,实际上,我们也不需要有多份拷贝。

为了解决这一问题,我们让类C1和类C2在继承B的时候进行了虚继承,即在继承方式前加virtual ,虚基类使得在继承间接共同基类时只保留一份成员。

数据结果分析:

在代码中,我们只给类C1中继承的b赋值为3,但是截图显示,类C1和类C2中的数据成员b的值都为3,而且类C1和类C2 中数据成员Bde地址是一样的,这就验证了“虚基类使得在继承间接共同基类时只保留一份成员

虚基类并不是在声明基类时声明的,而是在声明派生类时,指定继承方式时声明的。

声明虚基类的一般方式为:

class 派生类名:virtual 继承方式 基类名

在最后的派生类中,不仅要负责对其直接基类进行初始化,还要负责对虚基类初始化


对于这个例子

有虚基类的时候:

wKioL1cYkkyDy4GDAAAKL_QIo0A258.png

没有虚基类的时候:

wKiom1cYkYrwCUt5AAAIH9RZuZU047.png

没有虚基类的时候,对于这种菱形继承,派生类D中会有两个数据成员int b,也是因此,在访问的时候必须要写清楚访问的是那个类里面的b成员,如C1::b,否则会因为访问成员不明确而造成错误,为了避免这一问题,要在声明继承方式的时候加上virtual关键字,表示这是一个虚继承


通过看内存,可以清楚的看到在类D中,最后继承下来的数据成员是如何放置的

虚基类_第3张图片

虚基类_第4张图片

c1中的虚基类地址存储的偏移量20

wKiom1cYk5ijf_GjAAAF-7H2awI724.png

c2中的虚基类地址存储的偏移量12

wKioL1cYlFqCvby1AAAF9RtPuB4511.png

在内存中大概就是这个样子:

虚基类_第5张图片


你可能感兴趣的:(虚基类)