C++ 深入理解 虚继承、多重继承和直接继承
【摘要】
本文从5段代码实例出发,通过类中类的普通继承,类的虚继承,类的多重继承,多个虚函数类的普通继承、虚继承与多重继承,几个交叉概念,详细的阐释了继承、虚函数与虚继承的基本概念,深入剖析了继承于虚继承的区别于联系。
【Exp.001-虚继承】
- #include <stdio.h>
- class A {
- public:
- int a;
- };//sizeof(A)=4
- class B : virtual public A {
- public:
- int b;
- };//sizeof(B)=4(A副本)+4(虚表指针)+4(自己变量)=12
- class C : virtual public B {
- };//sizeof(c)= 12(B副本)+4(虚表指针) = 16,如果这里改为直接继承,那么sizeof(c)=12
- int main() {
- printf("%d\n", sizeof(C));
- return ;
- }
解析:这里需要理解虚继承基类对派生类的空间大小的影响,理解虚指针在虚继承中为子类带来了哪些空间的改变。
【Exp.002-多重继承】
- #include <stdio.h>
- class A {
- public:
- int a;
- };//sizeof(A) = 4
- class B : virtual public A {
- };// sizeof(B) =4+4=8
- class C : virtual public A {
- };//sizeof(C) =4+4=8
- class D : public B, public C{
- };
- //sizeof(D)=8+8-4=12 这里需要注意要减去4,因为B和C同时继承A,只需要保存一个A的副本就好了,sizeof(D)=4(A的副本)+4(B的虚表)+4(C的虚表)=12
- int main() {
- printf("%d\n", sizeof(D));
- return ;
- }
解析:这里需要关注 class D 的数据空间大小,理解多重虚继承对派生类虚指针以及派生类空间的影响。
【Exp.003-普通继承(含有:空类、虚函数)】
- class A
- {
- };
- class B
- {
- char ch;
- virtual void func0() { }
- };
- class C
- {
- char ch1;
- char ch2;
- virtual void func() { }
- virtual void func1() { }
- };
- class D: public A, public C
- {
- int d;
- virtual void func() { }
- virtual void func1() { }
- };
- class E: public B, public C
- {
- int e;
- virtual void func0() { }
- virtual void func1() { }
- };
- int main(void)
- {
- cout<<"A="<<sizeof(A)<<endl;//result=1 空类所占空间的大小为 1
- cout<<"B="<<sizeof(B)<<endl;//result=8 1+4 对齐 8
- cout<<"C="<<sizeof(C)<<endl;//result=8 1+1+4 对齐 8
- cout<<"D="<<sizeof(D)<<endl;//result=12 C的副本+D本身=8+4=12
- cout<<"E="<<sizeof(E)<<endl;//result=20 B的副本+C的副本+E本身=8+8+4=20
- return ;
- }
这里需要区分一下:
①没有继承的时候,存在虚函数则需要加上虚指针,如果有多个也只需要加上一个,因为只有一个虚指针;
②对于普通继承,类D和类E中自己的虚函数,大小为0,因为,它没有虚表;
③对于虚继承中,派生类中存在一个或多个虚函数的时候,它本身就有一个虚表,指向自己的虚表,所以要加4。
【Exp.004-虚继承(多重继承和虚函数)】
- class CommonBase
- {
- int co;
- };// size = 4
- class Base1: virtual public CommonBase
- {
- public:
- virtual void print1() { }
- virtual void print2() { }
- private:
- int b1;
- };//4副本+4虚指针+4自身+4(虚继承+虚函数构成指针多一个)=16
- class Base2: virtual public CommonBase
- {
- public:
- virtual void dump1() { }
- virtual void dump2() { }
- private:
- int b2;
- };//同理16
- class Derived: public Base1, public Base2
- {
- public:
- void print2() { }
- void dump2() { }
- private:
- int d;
- };//16+16-4+4=32
解析:如果不是虚继承的类,即便有虚函数也不会因此增加存储空间,如果是虚继承的类,没有虚函数就添加一个虚指针空间,有虚函数不论多少个,就添加两个虚指针空间。
【Exp.005-虚继承与虚函数】
- class A
- {
- public:
- virtual void aa() { }
- virtual void aa2() { }
- private:
- char ch[];
- }; // 1+4 = 补齐 = 8
- class B: virtual public A
- {
- public:
- virtual void bb() { }
- virtual void bb2() { }
- }; // 8(副本)+4(虚继承)+4(虚指针) = 16
- int main(void)
- {
- cout<<"A's size is "<<sizeof(A)<<endl;// 4+4=8
- cout<<"B's size is "<<sizeof(B)<<endl;// A的副本+4+4=16
- return ;
- }
解析:如果不是虚继承的类,即便有虚函数也不会因此增加存储空间,如果是虚继承的类,没有虚函数就添加一个虚指针空间,有虚函数不论多少个,就添加两个虚指针空间。
【小结】
重要的事情讲三遍!!!
如果不是虚继承的类,即便有虚函数也不会因此增加存储空间,如果是虚继承的类,没有虚函数就添加一个虚指针空间,有虚函数不论多少个,就添加两个虚指针空间!!!
原文地址:https://blog.csdn.net/u013630349/article/details/47057929
C++ 深入理解 虚继承、多重继承和直接继承的更多相关文章
- C++_day8_ 多重继承、钻石继承和虚继承
1.继承的复习 1.1 类型转换 编译器认为访问范围缩小是安全的. 1.2 子类的构造与析构 子类中对基类构造函数初始化只能写在初始化表里,不能写在函数体中. 阻断继承. 1.3 子类的拷贝构造与拷贝 ...
- C++多重继承与虚拟继承
本文只是粗浅讨论一下C++中的多重继承和虚拟继承. 多重继承中的构造函数和析构函数调用次序 我们先来看一下简单的例子: #include <iostream> using namespac ...
- 图文例解C++类的多重继承与虚拟继承
文章导读:C++允许为一个派生类指定多个基类,这样的继承结构被称做多重继承. 在过去的学习中,我们始终接触的单个类的继承,但是在现实生活中,一些新事物往往会拥有两个或者两个以上事物的属性,为了解决这个 ...
- c++继承汇总(单继承、多继承、虚继承、菱形继承)
多重继承中,一个基类可以在派生层次中出现多次,如果一个派生类有多个直接基类,而这些直接基类又有一个共同的基类,则在最终的派生类中会保留该间接共同基类数据成员的多分同名成员.C++提供虚基类的方法使得在 ...
- 《挑战30天C++入门极限》图文例解C++类的多重继承与虚拟继承
图文例解C++类的多重继承与虚拟继承 在过去的学习中,我们始终接触的单个类的继承,但是在现实生活中,一些新事物往往会拥有两个或者两个以上事物的属性,为了解决这个问题,C++引入了多重继承的概念 ...
- C++逆向分析----多重继承和菱形继承
多重继承 多重继承是指C++类同时继承两个类或两个以上的类. class Test { public: int num1; Test() { num1 = 1; } virtual void Proc ...
- java提高篇(二)-----理解java的三大特性之继承
在<Think in java>中有这样一句话:复用代码是Java众多引人注目的功能之一.但要想成为极具革命性的语言,仅仅能够复制代码并对加以改变是不够的,它还必须能够做更多的事情.在这句 ...
- javascript中继承(二)-----借用构造函数继承的个人理解
本人目录如下: 零.寒暄&回顾 一,借用构造函数 二.事件代理 三,call和apply的用法 四.总结 零.寒暄&回顾 上次博客跟大家分享了自己对原型链继承的理解,想看的同学欢迎猛击 ...
- 对Java不能多继承,只能单继承,却可以实现多个接口的理解
1.java与C++的不同点在于多继承. Java:不能多继承,只能单继承,但可以实现多个接口 C++:可以实现多继承.例如: class A extends B implements C,D,E { ...
随机推荐
- 廖雪峰Java9正则表达式-2正则表达式进阶-3分组匹配
1.使用括号可以提取字符串 不加括号匹配电话号码 匹配成功后,如何提取想要的字符串? 使用(...)可以分组:"^(\d{3,4})\-(\d{6,8})$" 2.String.m ...
- MFC如何添加背景图片
1.在xxDlg.h中定义CBrush m_brush 2.在对话框的消息响应中添加一个ON_WM_CTLCOLOR消息响应 3.添加背景图(方法前一篇随笔有讲到) 4.OnCtlColor消息响应函 ...
- Intent之跳转总结
) { localIntent.setAction(; ActivityManager am = (ActivityManager) context.) {) { ) { // android 5.0 ...
- Qt文件系统之QFile
QFile文件操作 文件打开方式: QIODevice::NotOpen 0x0000 设备不打开.QIODevice::ReadOnly 0x0001 设备 以只读的方式打开.Q ...
- JAVA除法保留小数点后两位的两种方法
1.(double) (Math.round(sd3*10000)/10000.0); 这样为保持4位 (double) (Math.round(sd3*100)/100.0); 这样为保持2位 ...
- Ping 笔记
Ping 笔记 查看网络间的连通性. 当设备与PC在同一局域网中(连接同一路由器),设备与PC互Ping对方,若只有其中一个掉线连不上,两者可能存在网络端口的问题, 若设备与PC同时掉线,则为路由器 ...
- day73 Django框架之URL
Django框架之url路由层一 Django数据库的一对多与多对多表的建立 一对多 publish_id的建立:publish=models.ForeignKey(to='Publish', to ...
- Python笔记:编码问题
1. python2的编码: python2中使用的是ASCII码,所以不支持中文,如果要在python2中写入中文编码,需要在文件头编写: #-*- encoding:utf-8 -*- 2. 不同 ...
- python列表中的pop函数
再python的列表中,有许多的内置方法,而在这里我主要向大家介绍一下pop函数. pop函数主要是用于删除列表中的数据.而其删除值时会返回删除的值.如果没有参数传入时, 则会默认认为删除列表的最后一 ...
- MM-物料分类帐差异分摊
物料分类帐差异分摊-MR21-MR22: https://wenku.baidu.com/view/6599e8f326fff705cc170ad8.html