c++内存对齐原理
转载自http://blog.csdn.net/it_yuan/article/details/24651347
#类中的元素
0. 成员变量 1. 成员函数 2. 静态成员变量 3. 静态成员函数 4. 虚函数 5. 纯虚函数
#影响对象大小的因素
0. 成员变量 1. 虚函数表指针(_vftptr) 2. 虚基类表指针(_vbtptr) 3. 内存对齐
_vftptr、_vbtptr的初始化由对象的构造函数, 赋值运算符自动完成;对象生命周期结束后,由对象的析构函数来销毁。
对象所关联的类型(type_info),通常放在virtual table的第一个slot中。
虚继承:在继承定义中包含了virtual关键字的继承关系;
虚基类:在虚继承体系中的通过virtual继承而来的基类,需要注意的是:
class CDerive : public virtual CBase {}; 其中CBase称之为CDerive的虚基类,而不是说CBase就是个虚基类,因为CBase还可以为不是虚继承体系中的基类。
虚函数被派生后,仍然为虚函数,即使在派生类中省去virtual关键字。
注:【下文中_vbptr即_vbtptr】
#对象内存布局分类讨论
vc6变量查看器中(Locals,Watch1等),也可以看到部分对象布局的情况(不完整,且虚继承是错误的)。
vs2005及以后版本的编译器提供了/d1reportSingleClassLayout[类名]编译选项来查看对象完整的内存布局:
cl classLayout.cpp /d1reportSingleClassLayoutCChildren
0. 单一类
(1). 空类
sizeof(CNull)=1(用于标识该对象)
(2). 只有成员变量的类
int nVarSize = sizeof(CVariable) = 12
内存布局:
(3). 只有虚函数的类
int nVFuntionSize = sizeof(CVFuction) = 4(虚表指针)
内存布局:
(4). 有成员变量、虚函数的类
int nParentSize = sizeof(CParent) = 8
内存布局:
1. 单一继承(含成员变量、虚函数、虚函数覆盖)
int nChildSize = sizeof(CChildren) = 12
vc中显示的结果(注:还有1个虚函数CChildren::g1没有被显示出来):
d1reportSingleClass查看:
内存布局:
2. 多继承 (含成员变量、虚函数、虚函数覆盖)
int nChildSize = sizeof(CChildren) = 20
vc中显示的结果(注:还有2个虚函数CChildren::f2,CChildren::h2没有被显示出来,this指针的adjustor[调整值]也没打印出):
d1reportSingleClass查看:
内存布局:
3. 深度为2的继承(含成员变量、虚函数、虚函数覆盖)
int nGrandSize = sizeof(CGrandChildren) = 24
vc中显示的结果(注:还有3个虚函数CGrandChildren::f2,CChildren::h2,CGrandChildren::f3没有显示出来,this指针的adjustor[调整值]也没打印出):
d1reportSingleClass查看:
内存布局:
4 重复继承(含成员变量、虚函数、虚函数覆盖)
int nGrandSize = sizeof(CGrandChildren) = 28
vc中显示的结果(注:还有大量的虚函数没有显示出来,this指针的adjustor[调整值]也没打印出):
thunk函数:一种形实转换辅助函数;主要做this指针调整,函数调用重定向。
d1reportSingleClass查看:
内存布局:
由于m_nAge在内容中存在两个拷贝,因此我们不能直接通过pGrandChildrenA->m_nAge来访问该变量,
这样会存在二义性,编译器无法知道应该访问CChildren1中的m_nAge,还是CChildren2中的m_nAge。
为了标识唯一的m_nAge,就需要带上其所在范围的类名了。如下:
1 pGrandChildrenA->CChildren1::m_nAge = 1; 2 pGrandChildrenA->CChildren2::m_nAge = 2;
5. 单一虚继承(含成员变量、虚函数、虚函数覆盖)
int nChildSize = sizeof(CChildren) = 20
d1reportSingleClass查看:
内存布局:
6. 多虚继承(含成员变量、虚函数、虚函数覆盖)
(1) virtual CParent1, CParent2
int nChildSize = sizeof(CChildren) = 24
d1reportSingleClass查看:
内存布局:
(2) CParent1, virtual CParent2
int nChildSize = sizeof(CChildren) = 24
d1reportSingleClass查看:
内存布局:
(3) virtual CParent1, virtual CParent2
int nChildSize = sizeof(CChildren) = 28
d1reportSingleClass查看:
内存布局:
7. 钻石型的虚拟多重继承(含成员变量、虚函数、虚函数覆盖)
int nGrandChildSize = sizeof(CGrandChildren) = 36
d1reportSingleClass查看:
thunk函数:一种形实转换辅助函数;主要做this指针调整,函数调用重定向。
内存布局:
#外部参考
参考链接:http://blog.csdn.net/yangshuangtao/article/details/45192149
相关文章
- c++中基类与派生类中隐含的this指针的分析
- 【C/C++开发】内存对齐(内存中的数据对齐)、大端模式及小端模式
- 【VS开发】【C/C++开发】传递双重指针申请内存,典型用法
- 【VS开发】【编程开发】【C/C++开发】结构体中的数组与指针的内存分配情况说明
- 【C++】继承/多态/文件
- C/C++:指针、引用【指针:指针是一个变量,只不过这个变量中存储的是一个地址,指向内存中的一个单元】【引用:引用和原变量是同一个东西,只不过是原变量的一个别名】
- C++内存管理5-处理new分配内存失败情况(转)
- C++内存管理3-探讨C++内存和回收
- LibSVM笔记系列(3)——初学移植libsvm的C/C++版本
- C++将文件内容一次性读入内存
- C++——堆、栈与内存管理
- [C++]单源最短路径:迪杰斯特拉(Dijkstra)算法(贪心算法)
- 高精度色子 投子 骰子定位识别C++/Python/C# opencv emgucv
- c++实验2(1.重载的函数 2.const关键字 3.动态内存分配利用new运算分配内存空间,利用delete运算或程序运行结束释放内存。)
- C++中子类出现与父类同名成员函数如果调用父类函数
- 《挑战30天C++入门极限》C/C++中字符指针数组及指向指针的指针的含义
- 【C/C++】C语言嵌入式编程修炼·背景篇·软件架构篇·内存操作篇
- C++客户端访问Java服务端发布的SOAP模式的WebService接口
- Blitz++与MTL两大数值计算程序库(C++)的简介
- C++类对应的内存结构
- 几个C++内存泄漏和越界检测工具简介
- C/C++获取操作系统、CPU、内存信息(windows和linux)
- C++著名类库和C++标准库介绍
- C++类对应的内存结构
- 25.C++- 泛型编程之函数模板(详解)
- 【C++11 多线程】仔细地将参数传递给线程(三)
- C++windows内核编程笔记day07_day08,可视化建菜单、加速键使用、绘图等