虚基类

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;

}

调试截图为:

原理分析:

因为类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 继承方式 基类名

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

对于这个例子

有虚基类的时候:

没有虚基类的时候:

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

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

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

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

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

时间: 2024-08-11 05:32:36

虚基类的相关文章

C++中虚基类

摘自<C++程序设计> 如果一个派生类有多个直接基类,而这些直接基类又有一个共同的基类,则在最终的派生类中会保留该间接共同基类数据成员的多份同名成员. C++提供虚基类(virtual base class)的方法,使得在继承间接共同基类时只保留一份成员. 下面举例说明: 在如下的图中: Person类是Student和Teacher的基类,而Graduate类又继承自Student和Teacher类. 如果使用虚基类的话,Graduate将有两份age拷贝,两份gender拷贝,两份name

C++的虚基类知识点

当在多条继承路径上有一个公共的基类,在这些路径的某几条汇合处,这个公共的基类就会产生多个实例(或多个副本),若只想保存这个基类的一个实例,可以将这个公共基类说明为虚基类. class x1:virtual public x{//... ...};class x2:virtual public x{//... ...};虚基类的初始化 虚基类(虚拟继承)的初始化与一般多继承的初始化在语法上是一样的,但构造函数的调用次序不同. 派生类的构造函数的调用次序有三个原则:(1)虚基类的构造函数在非虚基类之

第十二周 阅读项目 (4)虚基类多重继承数据理解

<span style="font-size:18px;">/* *Copyright (c)2014,烟台大学计算机与控制工程学院 *All rights reserved. *文件名称:d.cpp *作 者:张旺华 *完成日期:2015年6月1日 *版 本 号:v1.0 */ #include<iostream> using namespace std; class A { public: int n; }; class B:virtual public A

C++虚基类详解

1.虚基类的作用从上面的介绍可知:如果一个派生类有多个直接基类,而这些直接基类又有一个共同的基类,则在最终的派生类中会保留该间接共同基类数据成员的多份同名成员.在引用这些同名的成员时,必须在派生类对象名后增加直接基类名,以避免产生二义性,使其惟一地标识一个成员,如    c1.A::display( ).在一个类中保留间接共同基类的多份同名成员,这种现象是人们不希望出现的.C++提供虚基类(virtual base class )的方法,使得在继承间接共同基类时只保留一份成员.现在,将类A声明为

虚基类练习:动物(利用虚基类建立一个类的多重继承,包括动物(animal,属性有体长,体重和性别),陆生动物(ter_animal,属性增加了奔跑速度),水生动物(aqu_animal,属性增加了游泳速度)和两栖动物(amp_animal)。)

Description 长期的物种进化使两栖动物既能活跃在陆地上,又能游动于水中.利用虚基类建立一个类的多重继承,包括动物(animal,属性有体长,体重和性别),陆生动物(ter_animal,属性增加了奔跑速度),水生动物(aqu_animal,属性增加了游泳速度)和两栖动物(amp_animal).其中两栖动物保留了陆生动物和水生动物的属性. Input 两栖动物的体长,体重,性别,游泳速度,奔跑速度(running_speed) Output 初始化的两栖动物的体长,体重,性别,游泳速度

C++:虚基类

4.4.3 虚基类1.没什么要引入虚基类 如果一个类有多个直接基类,而这些直接基类又有一个共同的基类,则在最底层的派生类中会保留这个间接的共同基类数据成员的多分同名成员.在访问这些同名的成员时,必须在派生类对象后增加直接基类名,使其惟一地标识一个成员,以免产生二义性. //例 4.15 虚基类的引例 #include<iostream> using namespace std; class Base{ //声明类Base1和类Base2的共同的基类Base public: Base() { a

C++中 引入虚基类的作用

当某类的部分或全部直接基类是从另一个基类共同派生而来时,这直接基类中,从上一级基类继承来的成员就拥有相同的名称,派生类的对象的这些同名成员在内存中同时拥有多个拷贝,同一个函数名有多个映射.可以使用作用域分辨符来唯一标识并分别访问它们.也可以将共同基类设置为虚基类,这时从不同的路径继承过来的同名数据成员在内存中只拥有一个拷贝,同一个函数名也只有一个映射.也就是说虚基类解决了同名成员的唯一标识问题.

虚基类的用法

1 #include <iostream> 2 3 using namespace std; 4 5 class A 6 { 7 private: 8 int a; 9 public: 10 A(int x):a(x){} 11 void show() const 12 { 13 cout<<"a: "<<a<<endl; 14 } 15 ~A(){} 16 }; 17 class B:virtual public A //定义虚基类的用

虚基类初始化问题

在包含有继承关系的类里,生成一个派生类对象,要调用构造函数进行初始化此对象,而构造函数的调用顺序是先调用最顶层基类的构造函数,次顶层....等:但在普通继承和虚继承里存在区别 普通继承:父类只能由其直接派生类初始化 1 class A 2 { 3 char a; 4 public: 5 A(char _a) :a(_a){ cout << a << endl; } 6 }; 7 8 class B:public A 9 { 10 char b; 11 public: 12 B(ch

一目了然c++虚基类!

1 #include <IOSTREAM.H> 2 //基类 3 class CBase 4 ...{ 5 protected: 6 int a; 7 public: 8 CBase(int na) 9 ...{ 10 a=na; 11 cout<<"CBase constructor! "; 12 } 13 14 ~CBase()...{cout<<"CBase deconstructor! ";} 15 }; 16 17 //