C++ 虚函数和多重继承的内存布局初探
C++ 对象的内存布局
一切以事实说话:
代码:
1: #include <stdio.h>
2:
3: class A {
4: public:
5: int a;
6: int b;
7: int c;
8: };
9:
10: int main(int argc, char** argv) {
11: A obj;
12: printf(" obj :%p\n obj.a: %p \n obj.b: %p \n obj.c: %p\n" ,
13: &obj, &obj.a, &obj.b, &obj.c);
14: return 0;
15: }
执行结果:

不同的机器执行的结果可能不同,但是从这里的出的一个结论是:
对象的地址是整个类的起始地址(也就是低地址)。其成员的地址顺序和其在类中声明的顺序有关系。
而对上面的代码稍加修改,增加一个虚函数。
1: #include <stdio.h>
2:
3: class A {
4: public:
5: virtual void show() {}
6: int a;
7: int b;
8: int c;
9: };
10:
11: int main(int argc, char** argv) {
12: A obj;
13: printf(" obj :%p\n obj.a: %p \n obj.b: %p \n obj.c: %p\n" ,
14: &obj, &obj.a, &obj.b, &obj.c);
15: return 0;
16: }
测试结果:

这里是由于增加了一个虚函数表的指针(测试机器为64位系统,故指针为8个字节)。从这里可以看出,虚函数表指针在类的起始地址,这也是为了对于不同的类该地址的偏移相同。
1: class X {
2: };
3: class Y {
4: public:
5: virtual void f() {};
6: };
7: class X1 : public virtual X {
8: };
9: class X2 : public virtual X {
10: };
11: class A1 : public X1, public X2 {
12: };
13: class X3 : public X {
14: };
15: class X4 : public X {
16: };
17: class A2 : public X3, public X4 {
18: };
19: class Y1 : public virtual Y {
20: };
21: class Y2 : public virtual Y {
22: };
23: class B1 : public Y1, public Y2 {
24: };
25: class Y3 : public Y {
26: };
27: class Y4 : public Y {
28: };
29: class B2 : public Y3, public Y4 {
30: };
31:
32: int main (int argc, char** argv) {
33: printf("sizeof(X) %lu\n", sizeof(X));
34: printf("sizeof(Y) %lu\n", sizeof(Y));
35: printf("sizeof(X1) %lu\n", sizeof(X1));
36: printf("sizeof(X2) %lu\n", sizeof(X2));
37: printf("sizeof(A1) %lu\n", sizeof(A1));
38: printf("sizeof(X3) %lu\n", sizeof(X3));
39: printf("sizeof(X4) %lu\n", sizeof(X4));
40: printf("sizeof(A2) %lu\n", sizeof(A2));
41: printf("sizeof(Y1) %lu\n", sizeof(Y1));
42: printf("sizeof(Y2) %lu\n", sizeof(Y2));
43: printf("sizeof(B1) %lu\n", sizeof(B1));
44: printf("sizeof(Y3) %lu\n", sizeof(Y3));
45: printf("sizeof(Y4) %lu\n", sizeof(Y4));
46: printf("sizeof(B2) %lu\n", sizeof(B2));
47: return 0;
48: }
执行结果:

上面的测试结果得出的结论:
- 空类的大小不是0,而是1,这样是提供一个占位符,这样由空类创建出的两个对象的地址就会不同,以作区分。
- 当类不是空的时候,该占位符就不需要了,类的大小就是类中成员所需的空间大小
注: 本想深入一探究竟,不过这部分和编译器有很大的关系,非朝夕可以搞定。先写这么多,再多做了解后再补充吧。
refer:
http://blog.csdn.net/haoel/article/details/3081328
http://blog.csdn.net/haoel/article/details/3081385
http://blog.chinaunix.net/uid-22535463-id-2749544.html
C++ 虚函数和多重继承的内存布局初探的更多相关文章
- vs查看虚函数表和类内存布局
虚继承和虚基类 虚继承:在继承定义中包含了virtual关键字的继承关系: 虚基类:在虚继承体系中的通过virtual继承而来的基类,需要注意的是:class CSubClass : publ ...
- 如何使用C#调用C++类虚函数(即动态内存调用)
本文讲解如何使用C#调用只有.h头文件的c++类的虚函数(非实例函数,因为非虚函数不存在于虚函数表,无法通过类对象偏移计算地址,除非用export导出,而gcc默认是全部导出实例函数,这也是为什么 ...
- C++中虚函数继承类的内存占用大小计算
计算一个类对象的大小时的规律: 1.空类.单一继承的空类.多重继承的空类所占空间大小为:1(字节,下同): 2.一个类中,虚函数本身.成员函数(包括静态与非静态)和静态数据成员都是不占用类对象的存储空 ...
- 从零开始学C++之虚继承和虚函数对C++对象内存模型造成的影响
首先重新回顾一下关于类/对象大小的计算原则: 类大小计算遵循结构体对齐原则 第一个数据成员放在offset为0的位置 其它成员对齐至min(sizeof(member),#pragma pack(n) ...
- Vc++内存布局
Vc++内存布局 测试平台 Windows server 2012 R2 and visual studio 2013 professional. 本篇文章意在介绍vc++中类的内存布局方式,只是研究 ...
- HotSpot源码分析之C++对象的内存布局
HotSpot采用了OOP-Klass模型来描述Java类和对象.OOP(Ordinary Object Pointer)指的是普通对象指针,而Klass用来描述对象的具体类型.为了更好理解这个模型, ...
- c++ 对象内存分配和虚函数
1. c++类对象(不含虚函数)在内存中的分布 c++类中有四种成员:静态数据.非静态数据.静态函数.非静态函数. 1. 非静态数据成员放在每个对象内部,作为对象专有的数据成员 2. 静态数据成员被抽 ...
- 浅析GCC下C++多重继承 & 虚拟继承的对象内存布局
继承是C++作为OOD程序设计语言的三大特征(封装,继承,多态)之一,单一非多态继承是比较好理解的,本文主要讲解GCC环境下的多重继承和虚拟继承的对象内存布局. 一.多重继承 先看几个类的定义: 01 ...
- 虚函数列表: 取出方法 // 虚函数工作原理和(虚)继承类的内存占用大小计算 32位机器上 sizeof(void *) // 4byte
#include <iostream> using namespace std; class A { public: A(){} virtual void geta(){ cout < ...
随机推荐
- 日常Git使用——2019年12月11日16:19:03
1.git介绍 1.1 什么是git? 什么是Git? 比如一个项目,两个人同时参与开发,那么就把这个项目放在一个公共的地方,需要的时候都可以去获取,有什么改动,都可以进行提交. 为了做到这一点,就需 ...
- 大数据之hadoop框架知识
https://blog.csdn.net/zytbft/article/details/79285500
- 分组函数 partition by 的详解,与order by 区别
partition by关键字是分析性函数的一部分,它和聚合函数(如group by)不同的地方在于它能返回一个分组中的多条记录,而聚合函数一般只有一条反映统计值的记录, partition by ...
- mysql 8.0版本下载安装以及默认密码修改
1. 下载 去mysql官网下载地址进行下载,选择需要的安装包 可以直接跳过登录进行下载 ps:我是先注册账号下载的,注册时遇到一个坑,就是在chrome75版本无法选择省市,不选择又不让注册. 可以 ...
- Codeforces 831C--Jury Marks (思维)
题目链接:http://codeforces.com/problemset/problem/831/C 题意:有一位参赛选手,我们不知道他初始或最后的成绩,但是知道k次评审所加(减)的分数,以及n个在 ...
- php基于SQLite实现的分页功能示例
php基于SQLite实现的分页功能. 这里操作数据库文件使用的是前面文章<PHP基于PDO实现的SQLite操作类>中的SQLite数据库操作类. 代码: <?php class ...
- 2019 牛客暑期多校 第八场 A All-one Matrices (单调栈+前缀和)
题目:https://ac.nowcoder.com/acm/contest/888/A 题意:找全1矩阵的个数,并且这个全1矩阵不被其他全1矩阵包含 思路:这里引用付队说的话 -> { 这类问 ...
- [HDU2855]Fibonacci Check-up
题目:Fibonacci Check-up 链接:http://acm.hdu.edu.cn/showproblem.php?pid=2855 分析: 1)二项式展开:$(x+1)^n = \sum^ ...
- [CSP-S模拟测试86]题解
好久没有写整套题的题解了呢……主要是这两天考试题愈发神仙 实在是超出了垃圾博主的能力范围啊QAQ A.异或 不难想到,如果我们得到了$[L,R]$中每一位上0和1的个数,那么答案即为$2 \times ...
- python and 用法
>>> 1 and [] and [1] [] >>> 1 and [2] and [1] [1] >>> 0 and [1] and [2] 0