c++内存布局最完整
来源:互联网 发布:中国医疗人工智能大会 编辑:程序博客网 时间:2024/05/16 04:47
大家都应该知道C++的精髓是虚函数吧? 虚函数带来的好处就是: 可以定义一个基类的指针, 其指向一个继承类, 当通过基类的指针去调用函数时, 可以在运行时决定该调用基类的函数还是继承类的函数. 虚函数是实现多态(动态绑定)/接口函数的基础. 可以说: 没有虚函数, C++将变得一无是处!
既然是C++的精髓, 那么我们有必要了解一下她的实现方式吗? 有必要! 既然C++是从C语言的基础上发展而来的, 那么我们可以尝试用C语言来模拟实现吗? 有可能! 接下来, 就是我一步一步地来解析C++的虚函数的实现方式, 以及用C语言对其进行的模拟.
C++对象的内存布局
要想知道C++对象的内存布局, 可以有多种方式, 比如:
- 输出成员变量的偏移, 通过offsetof宏来得到
- 通过调试器查看, 比如常用的VS
只有数据成员的对象
类实现如下:
class Base1{public: int base1_1; int base1_2;};
对象大小及偏移:
sizeof(Base1)8offsetof(Base1, base1_1)0offsetof(Base1, base1_2)4可知对象布局:
可以看到, 成员变量是按照定义的顺序来保存的, 最先声明的在最上边, 然后依次保存!
类对象的大小就是所有成员变量大小之和.没有虚函数的对象
类实现如下:
class Base1{public: int base1_1; int base1_2; void foo(){}};
结果如下:
sizeof(Base1)8offsetof(Base1, base1_1)0offsetof(Base1, base1_2)4和前面的结果是一样的? 不需要有什么疑问对吧?
因为如果一个函数不是虚函数,那么他就不可能会发生动态绑定,也就不会对对象的布局造成任何影响.
当调用一个非虚函数时, 那么调用的一定就是当前指针类型拥有的那个成员函数. 这种调用机制在编译时期就确定下来了.拥有仅一个虚函数的类对象
类实现如下:
class Base1{public: int base1_1; int base1_2; virtual void base1_fun1() {}};
结果如下:
sizeof(Base1)12offsetof(Base1, base1_1)4offsetof(Base1, base1_2)8咦? 多了4个字节? 且 base1_1 和 base1_2 的偏移都各自向后多了4个字节!
说明类对象的最前面被多加了4个字节的"东东", what's it?
现在, 我们通过VS2013来瞧瞧类Base1的变量b1的内存布局情况:
(由于我没有写构造函数, 所以变量的数据没有根据, 但虚函数是编译器为我们构造的, 数据正确!)
(Debug模式下, 未初始化的变量值为0xCCCCCCCC, 即:-858983460)看到没? base1_1前面多了一个变量 __vfptr(常说的虚函数表vtable指针), 其类型为void**, 这说明它是一个void*指针(注意:不是数组).
再看看[0]元素, 其类型为void*, 其值为 ConsoleApplication2.exe!Base1::base1_fun1(void), 这是什么意思呢? 如果对WinDbg比较熟悉, 那么应该知道这是一种惯用表示手法, 她就是指 Base1::base1_fun1() 函数的地址.
可得, __vfptr的定义伪代码大概如下:
void* __fun[1] = { &Base1::base1_fun1 };const void** __vfptr = &__fun[0];
值得注意的是:
- sizeof(Base1)12offsetof(__vfptr)0offsetof(base1_1)4offsetof(base1_2)8
我为什么要提这样一个问题? 因为如果仅是一个指针的情况, 您就无法轻易地修改那个数组里面的内容, 因为她并不属于类对象的一部分.
属于类对象的, 仅是一个指向虚函数表的一个指针__vfptr而已, 下一节我们将继续讨论这个问题.- 注意到__vfptr前面的const修饰. 她修饰的是那个虚函数表, 而不是__vfptr.
现在的对象布局如下:
虚函数指针__vfptr位于所有的成员变量之前定义.
注意到: 我并未在此说明__vfptr的具体指向, 只是说明了现在类对象的布局情况.
接下来看一个稍微复杂一点的情况, 我将清楚地描述虚函数表的构成.拥有多个虚函数的类对象
和前面一个例子差不多, 只是再加了一个虚函数. 定义如下:
class Base1{public: int base1_1; int base1_2; virtual void base1_fun1() {} virtual void base1_fun2() {}};
大小以及偏移信息如下:
有情况!? 多了一个虚函数, 类对象大小却依然是12个字节!
再来看看VS形象的表现:
呀, __vfptr所指向的函数指针数组中出现了第2个元素, 其值为Base1类的第2个虚函数base1_fun2()的函数地址.
现在, 虚函数指针以及虚函数表的伪定义大概如下:
void* __fun[] = { &Base1::base1_fun1, &Base1::base1_fun2 };const void** __vfptr = &__fun[0];
通过上面两张图表, 我们可以得到如下结论:
- 更加肯定前面我们所描述的: __vfptr只是一个指针, 她指向一个函数指针数组(即: 虚函数表)
- 增加一个虚函数, 只是简单地向该类对应的虚函数表中增加一项而已, 并不会影响到类对象的大小以及布局情况
前面已经提到过: __vfptr只是一个指针, 她指向一个数组, 并且: 这个数组没有包含到类定义内部, 那么她们之间是怎样一个关系呢?
不妨, 我们再定义一个类的变量b2, 现在再来看看__vfptr的指向:通过Watch 1窗口我们看到:
- b1和b2是类的两个变量, 理所当然, 她们的地址是不同的(见 &b1 和 &b2)
- 虽然b1和b2是类的两个变量, 但是: 她们的__vfptr的指向却是同一个虚函数表
由此我们可以总结出:
同一个类的不同实例共用同一份虚函数表, 她们都通过一个所谓的虚函数表指针__vfptr(定义为void**类型)指向该虚函数表.
是时候该展示一下类对象的内存布局情况了:
不出意外, 很清晰明了地展示出来了吧? :-) hoho~~
那么问题就来了! 这个虚函数表保存在哪里呢? 其实, 我们无需过分追究她位于哪里, 重点是:
- 她是编译器在编译时期为我们创建好的, 只存在一份
- 定义类对象时, 编译器自动将类对象的__vfptr指向这个虚函数表
单继承且本身不存在虚函数的继承类的内存布局
前面研究了那么多啦, 终于该到研究继承类了! 先研究单继承!
依然, 简单地定义一个继承类, 如下:
class Base1{public: int base1_1; int base1_2; virtual void base1_fun1() {} virtual void base1_fun2() {}};class Derive1 : public Base1{public: int derive1_1; int derive1_2;};
我们再来看看现在的内存布局(定义为Derive1 d1):
没错! 基类在上边, 继承类的成员在下边依次定义! 展开来看看:
经展开后来看, 前面部分完全就是Base1的东西: 虚函数表指针+成员变量定义.
并且, Base1的虚函数表的[0][1]两项还是其本身就拥有的函数: base1_fun1() 和 base1_fun2().现在类的布局情况应该是下面这样:
本身不存在虚函数(不严谨)但存在基类虚函数覆盖的单继承类的内存布局
标题`本身不存在虚函数`的说法有些不严谨, 我的意思是说: 除经过继承而得来的基类虚函数以外, 自身没有再定义其它的虚函数.
Ok, 既然存在基类虚函数覆盖, 那么来看看接下来的代码会产生何种影响:
class Base1{public: int base1_1; int base1_2; virtual void base1_fun1() {} virtual void base1_fun2() {}};class Derive1 : public Base1{public: int derive1_1; int derive1_2; // 覆盖基类函数 virtual void base1_fun1() {}};
可以看到, Derive1类 重写了Base1类的base1_fun1()函数, 也就是常说的虚函数覆盖. 现在是怎样布局的呢?
特别注意我高亮的那一行: 原本是Base1::base1_fun1(), 但由于继承类重写了基类Base1的此方法, 所以现在变成了Derive1::base1_fun1()!
那么, 无论是通过Derive1的指针还是Base1的指针来调用此方法, 调用的都将是被继承类重写后的那个方法(函数), 多态发生鸟!!!
那么新的布局图:
定义了基类没有的虚函数的单继承的类对象布局
说明一下: 由于前面一种情况只会造成覆盖基类虚函数表的指针, 所以接下来我不再同时讨论虚函数覆盖的情况.
继续贴代码:
class Base1{public: int base1_1; int base1_2; virtual void base1_fun1() {} virtual void base1_fun2() {}};class Derive1 : public Base1{public: int derive1_1; int derive1_2; virtual void derive1_fun1() {}};
和第5类不同的是多了一个自身定义的虚函数. 和第6类不同的是没有基类虚函数的覆盖.
咦, 有没有发现问题? 表面上看来几乎和第5种情况完全一样? 为嘛呢?
现在继承类明明定义了自身的虚函数, 但不见了??
那么, 来看看类对象的大小, 以及成员偏移情况吧:居然没有变化!!! 前面12个字节是Base1的, 有没有觉得很奇怪?
好吧, 既然表面上没办法了, 我们就只能从汇编入手了, 来看看调用derive1_fun1()时的代码:
Derive1 d1;Derive1* pd1 = &d1;pd1->derive1_fun1();
要注意: 我为什么使用指针的方式调用? 说明一下: 因为如果不使用指针调用, 虚函数调用是不会发生动态绑定的哦! 你若直接 d1.derive1_fun1();, 是不可能会发生动态绑定的, 但如果使用指针: pd1->derive1_fun1(); , 那么 pd1就无从知道她所指向的对象到底是Derive1 还是继承于Derive1的对象, 虽然这里我们并没有对象继承于Derive1, 但是她不得不这样做, 毕竟继承类不管你如何继承, 都不会影响到基类, 对吧?
; pd1->derive1_fun1();00825466 mov eax,dword ptr [pd1] 00825469 mov edx,dword ptr [eax] 0082546B mov esi,esp 0082546D mov ecx,dword ptr [pd1] 00825470 mov eax,dword ptr [edx+8] 00825473 call eax
汇编代码解释:
第2行: 由于pd1是指向d1的指针, 所以执行此句后 eax 就是d1的地址
第3行: 又因为Base1::__vfptr是Base1的第1个成员, 同时也是Derive1的第1个成员, 那么: &__vfptr == &d1, clear? 所以当执行完 mov edx, dword ptr[eax] 后, edx就得到了__vfptr的值, 也就是虚函数表的地址.
第5行: 由于是__thiscall调用, 所以把this保存到ecx中.
第6行: 一定要注意到那个 edx+8, 由于edx是虚函数表的地址, 那么 edx+8将是虚函数表的第3个元素, 也就是__vftable[2]!!!
第7行: 调用虚函数.结果:
- 现在我们应该知道内幕了! 继承类Derive1的虚函数表被加在基类的后面! 事实的确就是这样!
- 由于Base1只知道自己的两个虚函数索引[0][1], 所以就算在后面加上了[2], Base1根本不知情, 不会对她造成任何影响.
- 如果基类没有虚函数呢? 这个问题我们留到第9小节再来讨论!
最新的类对象布局表示:
多继承且存在虚函数覆盖同时又存在自身定义的虚函数的类对象布局
真快, 该看看多继承了, 多继承很常见, 特别是接口类中!
依然写点小类玩玩:
class Base1{public: int base1_1; int base1_2; virtual void base1_fun1() {} virtual void base1_fun2() {}};class Base2{public: int base2_1; int base2_2; virtual void base2_fun1() {} virtual void base2_fun2() {}};// 多继承class Derive1 : public Base1, public Base2{public: int derive1_1; int derive1_2; // 基类虚函数覆盖 virtual void base1_fun1() {} virtual void base2_fun2() {} // 自身定义的虚函数 virtual void derive1_fun1() {} virtual void derive1_fun2() {}};
代码变得越来越长啦! 为了代码结构清晰, 我尽量简化定义.
初步了解一下对象大小及偏移信息:
貌似, 若有所思? 不管, 来看看VS再想:
哇, 不摆了! 一丝不挂啊! :-)
结论:
- 按照基类的声明顺序, 基类的成员依次分布在继承中.
- 注意被我高亮的那两行, 已经发生了虚函数覆盖!
- 我们自己定义的虚函数呢? 怎么还是看不见?!
好吧, 继承反汇编, 这次的调用代码如下:
Derive1 d1;Derive1* pd1 = &d1;pd1->derive1_fun2();
反汇编代码如下:
; pd1->derive1_fun2();00995306 mov eax,dword ptr [pd1] 00995309 mov edx,dword ptr [eax] 0099530B mov esi,esp 0099530D mov ecx,dword ptr [pd1] 00995310 mov eax,dword ptr [edx+0Ch] 00995313 call eax
解释下, 其实差不多:
第2行: 取d1的地址
第3行: 取Base1::__vfptr的值!!
第6行: 0x0C, 也就是第4个元素(下标为[3])结论:
Derive1的虚函数表依然是保存到第1个拥有虚函数表的那个基类的后面的.
看看现在的类对象布局图:
如果第1个基类没有虚函数表呢? 进入第9节!
如果第1个直接基类没有虚函数(表)
这次的代码应该比上一个要稍微简单一些, 因为把第1个类的虚函数给去掉鸟!
class Base1{public: int base1_1; int base1_2;};class Base2{public: int base2_1; int base2_2; virtual void base2_fun1() {} virtual void base2_fun2() {}};// 多继承class Derive1 : public Base1, public Base2{public: int derive1_1; int derive1_2; // 自身定义的虚函数 virtual void derive1_fun1() {} virtual void derive1_fun2() {}};
来看看VS的布局:
这次相对前面一次的图来说还要简单啦! Base1已经没有虚函数表了! (真实情况并非完全这样, 请继续往下看!)
现在的大小及偏移情况: 注意: sizeof(Base1) == 8;
重点是看虚函数的位置, 进入函数调用(和前一次是一样的):
Derive1 d1;Derive1* pd1 = &d1;pd1->derive1_fun2();
反汇编调用代码:
; pd1->derive1_fun2();012E4BA6 mov eax,dword ptr [pd1] 012E4BA9 mov edx,dword ptr [eax] 012E4BAB mov esi,esp 012E4BAD mov ecx,dword ptr [pd1] 012E4BB0 mov eax,dword ptr [edx+0Ch] 012E4BB3 call eax
这段汇编代码和前面一个完全一样!, 那么问题就来了! Base1 已经没有虚函数表了, 为什么还是把b1的第1个元素当作__vfptr呢?
不难猜测: 当前的布局已经发生了变化, 有虚函数表的基类放在对象内存前面!? , 不过事实是否属实? 需要仔细斟酌.我们可以通过对基类成员变量求偏移来观察:
可以看到:
&d1==0x~d4&d1.Base1::__vfptr==0x~d4&d1.base2_1==0x~d8&d1.base2_2==0x~dc&d1.base1_1==0x~e0&d1.base1_2==0x~e4
所以不难验证: 我们前面的推断是正确的, 谁有虚函数表, 谁就放在前面!
现在类的布局情况:
那么, 如果两个基类都没有虚函数表呢?
What if 两个基类都没有虚函数表
代码如下:
class Base1{public: int base1_1; int base1_2;};class Base2{public: int base2_1; int base2_2;};// 多继承class Derive1 : public Base1, public Base2{public: int derive1_1; int derive1_2; // 自身定义的虚函数 virtual void derive1_fun1() {} virtual void derive1_fun2() {}};
前面吃了个亏, 现在先来看看VS的基本布局:
可以看到, 现在__vfptr已经独立出来了, 不再属于Base1和Base2!
看看求偏移情况:
Ok, 问题解决! 注意高亮的那两行, &d1==&d1.__vfptr, 说明虚函数始终在最前面!
不用再废话, 相信大家对这种情况已经有底了.
对象布局:
如果有三个基类: 虚函数表分别是有, 没有, 有!
这种情况其实已经无需再讨论了, 作为一个完结篇....
上代码:
class Base1{public: int base1_1; int base1_2; virtual void base1_fun1() {} virtual void base1_fun2() {}};class Base2{public: int base2_1; int base2_2;};class Base3{public: int base3_1; int base3_2; virtual void base3_fun1() {} virtual void base3_fun2() {}};// 多继承class Derive1 : public Base1, public Base2, public Base3{public: int derive1_1; int derive1_2; // 自身定义的虚函数 virtual void derive1_fun1() {} virtual void derive1_fun2() {}};
只需要看看偏移就行了:
只需知道: 谁有虚函数表, 谁就往前靠!
C++中父子对象指针间的转换与函数调用
讲了那么多布局方面的东东, 终于到了尾声, 好累呀!!!
通过前面的讲解内容, 大家至少应该明白了各类情况下类对象的内存布局了. 如果还不会.....呃..... !@#$%^&*
进入正题~
由于继承完全拥有父类的所有, 包括数据成员与虚函数表, 所以:把一个继承类强制转换为一个基类是完全可行的.
如果有一个Derive1的指针, 那么:
- 得到Base1的指针: Base1* pb1 = pd1;
- 得到Base2的指针: Base2* pb2 = pd1;
- 得到Base3的指针: Base3* pb3 = pd1;
非常值得注意的是:
这是在基类与继承类之间的转换, 这种转换会自动计算偏移! 按照前面的布局方式!
也就是说: 在这里极有可能: pb1 != pb2 != pb3 ~~, 不要以为她们都等于 pd1!
至于函数调用, 我想, 不用说大家应该知道了:
- 如果不是虚函数, 直接调用指针对应的基本类的那个函数
- 如果是虚函数, 则查找虚函数表, 并进行后续的调用. 虚函数表在定义一个时, 编译器就为我们创建好了的. 所有的, 同一个类, 共用同一份虚函数表.
用C语言完全模拟C++虚函数表的实现与运作方式
如果对前面两大节的描述仔细了解了的话, 想用C语言来模拟C++的虚函数以及多态, 想必是轻而易举的事情鸟!
前提
但是, 话得说在前面, C++的编译器在生成类及对象的时候, 帮助我们完成了很多事件, 比如生成虚函数表!
但是, C语言编译器却没有, 因此, 很多事件我们必须手动来完成, 包括但不限于:
- 手动构造父子关系
- 手动创建虚函数表
- 手动设置__vfptr并指向虚函数表
- 手动填充虚函数表
- 若有虚函数覆盖, 还需手动修改函数指针
- 若要取得基类指针, 还需手动强制转换
- ......
总之, 要想用C语言来实现, 要写的代码绝对有点复杂.
C++原版调用
接下来, 我们都将以最后那个, 最繁杂的那个3个基类的实例来讲解, 但作了一些简化与改动:
- 用构造函数初始化成员变量
- 减少成员变量的个数
- 减少虚函数的个数
- 调用函数时产生相关输出
- Derive1增加一个基类虚函数覆盖
以下是对类的改动, 很少:
class Base1{public: Base1() : base1_1(11) {} int base1_1; virtual void base1_fun1() { std::cout << "Base1::base1_fun1()" << std::endl; }};class Base2{public: Base2() : base2_1(21) {} int base2_1;};class Base3{public: Base3() : base3_1(31) {} int base3_1; virtual void base3_fun1() { std::cout << "Base3::base3_fun1()" << std::endl; }};class Derive1 : public Base1, public Base2, public Base3{public: Derive1() : derive1_1(11) {} int derive1_1; virtual void base3_fun1() { std::cout << "Derive1::base3_fun1()" << std::endl; } virtual void derive1_fun1() { std::cout << "Derive1::derive1_fun1()" << std::endl; }};
为了看到多态的效果, 我们还需要定义一个函数来看效果:
void foo(Base1* pb1, Base2* pb2, Base3* pb3, Derive1* pd1){ std::cout << "Base1::\n" << " pb1->base1_1 = " << pb1->base1_1 << "\n" << " pb1->base1_fun1(): "; pb1->base1_fun1(); std::cout << "Base2::\n" << " pb2->base2_1 = " << pb2->base2_1 << std::endl; std::cout << "Base3::\n" << " pb3->base3_1 = " << pb3->base3_1 << "\n" <<" pb3->base3_fun1(): "; pb3->base3_fun1(); std::cout << "Derive1::\n" << " pd1->derive1_1 = " << pd1->derive1_1<< "\n" <<" pd1->derive1_fun1(): "; pd1->derive1_fun1(); std::cout<< " pd1->base3_fun1(): "; pd1->base3_fun1(); std::cout << std::endl;}
调用方式如下:
Derive1 d1;foo(&d1, &d1, &d1, &d1);
输出结果:
可以看到输出结果全部正确(当然了! :-), 哈哈~
同时注意到 pb3->base3_fun1() 的多态效果哦!
用C语言来模拟
必须要把前面的理解了, 才能看懂下面的代码!
为了有别于已经完成的C++的类, 我们分别在类前面加一个大写的C以示区分(平常大家都是习惯在C++写的类前面加C, 今天恰好反过来, 哈哈).
C语言无法实现的部分
C/C++是两个语言, 有些语言特性是C++专有的, 我们无法实现! 不过, 这里我是指调用约定, 我们应该把她排除在外.
对于类的成员函数, C++默认使用__thiscall, 也即this指针通过ecx传递, 这在C语言无法实现, 所以我们必须手动声明调用约定为:
- __stdcall, 就像微软的组件对象模型那样
- __cdecl, 本身就C语言的调用约定, 当然能使用了.
上面那种调用约定, 使用哪一种无关紧要, 反正不能使用__thiscall就行了.
因为使用了非__thiscall调用约定, 我们就必须手动传入this指针, 通过成员函数的第1个参数!
从最简单的开始: 实现 Base2
由于没有虚函数, 仅有成员变量, 这个当然是最好模拟的咯!
struct CBase2{ int base2_1;};
有了虚函数表的Base1, 但没被覆盖
下面是Base1的定义, 要复杂一点了, 多一个__vfptr:
struct CBase1{ void** __vfptr; int base1_1;};
因为有虚函数表, 所以还得单独为虚函数表创建一个结构体的哦!
但是, 为了更能清楚起见, 我并未定义前面所说的指针数组, 而是用一个包含一个或多个函数指针的结构体来表示!
因为数组能保存的是同一类的函数指针, 不太很友好!
但他们的效果是完全一样的, 希望读者能够理解明白!
struct CBase1_VFTable{ void(__stdcall* base1_fun1)(CBase1* that);};
注意: base1_fun1 在这里是一个指针变量!
注意: base1_fun1 有一个CBase1的指针, 因为我们不再使用__thiscall, 我们必须手动传入! Got it?
Base1的成员函数base1_fun1()我们也需要自己定义, 而且是定义成全局的:
void __stdcall base1_fun1(CBase1* that){ std::cout << "base1_fun1()" << std::endl;}
有虚函数覆盖的Base3
虚函数覆盖在这里并不能体现出来, 要在构造对象初始化的时候才会体现, 所以: base3其实和Base1是一样的.
struct CBase3{ void** __vfptr; int base3_1;};struct CBase3_VFTable{ void(__stdcall* base3_fun1)(CBase3* that);};
Base3的成员函数:
void __stdcall base3_fun1(CBase3* that){ std::cout << "base3_fun1()" << std::endl;}
定义继承类CDerive1
相对前面几个类来说, 这个类要显得稍微复杂一些了, 因为包含了前面几个类的内容:
struct CDerive1{ CBase1 base1; CBase3 base3; CBase2 base2; int derive1_1;};
特别注意: CBase123的顺序不能错!
另外: 由于Derive1本身还有虚函数表, 而且所以项是加到第一个虚函数表(CBase1)的后面的, 所以此时的CBase1::__vfptr不应该单单指向CBase1_VFTable, 而应该指向下面这个包含Derive1类虚函数表的结构体才行:
struct CBase1_CDerive1_VFTable{ void (__stdcall* base1_fun1)(CBase1* that); void(__stdcall* derive1_fun1)(CDerive1* that);};
因为CDerive1覆盖了CBase3的base3_fun1()函数, 所以不能直接用Base3的那个表:
struct CBase3_CDerive1_VFTable{ void(__stdcall* base3_fun1)(CDerive1* that);};
Derive1覆盖Base3::base3_fun1()的函数以及自身定义的derive1_fun1()函数:
void __stdcall base3_derive1_fun1(CDerive1* that){ std::cout << "base3_derive1_fun1()" << std::endl;}void __stdcall derive1_fun1(CDerive1* that){ std::cout << "derive1_fun1()" << std::endl;}
构造各类的全局虚函数表
由于没有了编译器的帮忙, 在定义一个类对象时, 所有的初始化工作都只能由我们自己来完成了!
首先构造全局的, 被同一个类共同使用的虚函数表!
// CBase1 的虚函数表CBase1_VFTable __vftable_base1;__vftable_base1.base1_fun1 = base1_fun1;// CBase3 的虚函数表CBase3_VFTable __vftable_base3;__vftable_base3.base3_fun1 = base3_fun1;
然后构造CDerive1和CBase1共同使用的虚函数表:
// CDerive1 和 CBase1 共用的虚函数表CBase1_CDerive1_VFTable __vftable_base1_derive1;__vftable_base1_derive1.base1_fun1 = base1_fun1;__vftable_base1_derive1.derive1_fun1 = derive1_fun1;
再构造CDerive1覆盖CBase3后的虚函数表: 注意: 数覆盖会替换原来的函数指针
CBase3_CDerive1_VFTable __vftable_base3_derive1;__vftable_base3_derive1.base3_fun1 = base3_derive1_fun1;
开始! 从CDerive1构造一个完整的Derive1类
先初始化成员变量与__vfptr的指向: 注意不是指错了!
CDerive1 d1;d1.derive1 = 1;d1.base1.base1_1 = 11;d1.base1.__vfptr = reinterpret_cast<void**>(&__vftable_base1_derive1); d1.base2.base2_1 = 21;d1.base3.base3_1 = 31;d1.base3.__vfptr = reinterpret_cast<void**>(&__vftable_base3_derive1);
由于目前的CDerive1是我们手动构造的, 不存在真正语法上的继承关系, 如要得到各基类指针, 我们就不能直接来取, 必须手动根据偏移计算:
char* p = reinterpret_cast<char*>(&d1);Base1* pb1 = reinterpret_cast<Base1*>(p + 0);Base2* pb2 = reinterpret_cast<Base2*>(p + sizeof(CBase1) + sizeof(CBase3));Base3* pb3 = reinterpret_cast<Base3*>(p + sizeof(CBase1));Derive1* pd1 = reinterpret_cast<Derive1*>(p);
真正调用:
foo(pb1, pb2, pb3, pd1);
调用结果:
结果相当正确!!!
2、虚继承:
原创作品,转载请标明:http://blog.csdn.net/xiejingfa/article/details/48028491
准备工作
1、VS2012使用命令行选项查看对象的内存布局
微软的Visual Studio提供给用户显示C++对象在内存中的布局的选项:/d1reportSingleClassLayout。使用方法很简单,直接在[工具(T)]选项下找到“Visual Studio命令提示(C)”后点击即可。切换到cpp文件所在目录下输入如下的命令即可
c1 [filename].cpp /d1reportSingleClassLayout[className]
其中[filename].cpp就是我们想要查看的class所在的cpp文件,[className]指我们想要查看的class的类名。(下面举例说明...)
2、查看普通多继承子类的内存布局
既然我们今天讲的是虚基类和虚继承,我们就先用上面介绍的命令提示工具查看一下普通多继承子类的内存布局,可以跟后文虚继承子类的内存布局情况加以比较。
我们新建一个名叫NormalInheritance的cpp文件,输入一下内容。
/** 普通继承(没有使用虚基类) */ // 基类A class A { public: int dataA; }; class B : public A { public: int dataB; }; class C : public A { public: int dataC; }; class D : public B, public C { public: int dataD; };
上面是一个简单的多继承例子,我们启动Visual Studio命令提示功能,切换到NormalInheritance.cpp文件所在目录,输入一下命令:
c1 NormalInheritance.cpp /d1reportSingleClassLayoutD
我们可以看到class D的内存布局如下:
从类D的内存布局可以看到A派生出B和C,B和C中分别包含A的成员。再由B和C派生出D,此时D包含了B和C的成员。这样D中就总共出现了2个A成员。大家注意到左边的几个数字,这几个数字表明了D中各成员在D中排列的起始地址,D中的五个成员变量(B::dataA、dataB、C::dataA、dataC、dataD)各占用4个字节,sizeof(D) = 20。
为了跟后文加以比较,我们再来看看B和C的内存布局:
虚继承的内存分布情况
上面我们看到了普通多继承子类的内存分布情况,下面我们进入主题,来看看典型的菱形虚继承子类的内存分布情况。
我们新建一个名叫VirtualInheritance的cpp文件,输入一
/** 虚继承(虚基类) */ #include <iostream> // 基类A class A { public: int dataA; }; class B : virtual public A { public: int dataB; }; class C : virtual public A { public: int dataC; }; class D : public B, public C { public: int dataD; };
VirtualInheritance.cpp和NormalInheritance.cpp的不同点在与C和C继承A时使用了virtual关键字,也就是虚继承。同样,我们看看B、C、D类的内存布局情况:
我们可以看到,菱形继承体系中的子类在内存布局上和普通多继承体系中的子类类有很大的不一样。对于类B和C,sizeof的值变成了12,除了包含类A的成员变量dataA外还多了一个指针vbptr,类D除了继承B、C各自的成员变量dataB、dataA和自己的成员变量外,还有两个分别属于B、C的指针。
那么类D对象的内存布局就变成如下的样子:
vbptr:继承自父类B中的指针
int dataB:继承自父类B的成员变量
vbptr:继承自父类C的指针
int dataC:继承自父类C的成员变量
int dataD:D自己的成员变量
int A:继承自父类A的成员变量
显然,虚继承之所以能够实现在多重派生子类中只保存一份共有基类的拷贝,关键在于vbptr指针。那vbptr到底指的是什么?又是如何实现虚继承的呢?其实上面的类D内存布局图中已经给出答案:
实际上,vbptr指的是虚基类表指针(virtual base table pointer),该指针指向了一个虚表(virtual table),虚表中记录了vbptr与本类的偏移地址;第二项是vbptr到共有基类元素之间的偏移量。在这个例子中,类B中的vbptr指向了虚表D::$vbtable@B@,虚表表明公共基类A的成员变量dataA距离类B开始处的位移为20,这样就找到了成员变量dataA,而虚继承也不用像普通多继承那样维持着公共基类的两份同样的拷贝,节省了存储空间。
为了进一步确定上面的想法是否正确,我们可以写一个简单的程序加以验证 + 4) << std:
int main() { D* d = new D; d->dataA = 10; d->dataB = 100; d->dataC = 1000; d->dataD = 10000; B* b = d; // 转化为基类B C* c = d; // 转化为基类C A* fromB = (B*) d; A* fromC = (C*) d; std::cout << "d address : " << d << std::endl; std::cout << "b address : " << b << std::endl; std::cout << "c address : " << c << std::endl; std::cout << "fromB address: " << fromB << std::endl; std::cout << "fromC address: " << fromC << std::endl; std::cout << std::endl; std::cout << "vbptr address: " << (int*)d << std::endl; std::cout << " [0] => " << *(int*)(*(int*)d) << std::endl; std::cout << " [1] => " << *(((int*)(*(int*)d)) + 1)<< std::endl; // 偏移量20 std::cout << "dataB value : " << *((int*)d + 1) << std::endl; std::cout << "vbptr address: " << ((int*)d + 2) << std::endl; std::cout << " [0] => " << *(int*)(*((int*)d + 2)) << std::endl; std::cout << " [1] => " << *((int*)(*((int*)d + 2)) + 1) << std::endl; // 偏移量12 std::cout << "dataC value : " << *((int*)d + 3) << std::endl; std::cout << "dataD value : " << *((int*)d + 4) << std::endl; std::cout << "dataA value : " << *((int*)d + 5) << std::endl; }
得到结果为:
3、虚继承基类中还有虚函数:
类的虚函数\虚继承所占的空间
本文地址: http://blog.csdn.net/caroline_wendy/article/details/24236469
char占用一个字节, 但不满足4的倍数时, 会进行数据对齐, 使用4的倍数的字节(byte).
GCC:
GCC中, 无论是虚函数还是虚继承, 都需要将指针存储在虚函数表(virtual function table), 占用4个字节.
继承会继承基类的数据, 和虚函数表, 即继承基类的空间.
代码:
[cpp] view plain copy print?/* * test.cpp * * Created on: 2014.04.20 * Author: Spike */ /*eclipse cdt, gcc 4.8.1*/ #include <iostream> using namespace std; class A { char k[5]; public: virtual void aa() {}; }; class B : virtual public A { char j[6]; public: virtual void bb() {}; }; class C : virtual public B { char i[7]; public: virtual void cc() {}; }; int main() { std::cout << "sizeof(A) = " << sizeof(A) << std::endl; std::cout << "sizeof(B) = " << sizeof(B) << std::endl; std::cout << "sizeof(C) = " << sizeof(C) << std::endl; return 0; } 输出:[plain] view plain copy print?sizeof(A) = 12 sizeof(B) = 24 sizeof(C) = 36
VS:
VS中, 正常情况下, 所有虚函数都写在一个虚函数表中, 即派生类使用基类的虚函数表, 不会新建字节的虚函数表;
如果使用虚继承, 则派生类, 新建虚函数表和虚继承表, 即额外增加8个字节(两个表);
输出:
- c++内存布局最完整
- c程序内存布局
- C/C++ 内存布局
- Objective-C内存布局
- C/C++ 内存布局
- Objective-C内存布局
- Objective-C内存布局
- C - 内存布局
- C语言内存布局
- C语言内存布局
- Objective-C内存布局
- C/C++ 内存布局
- Objective C内存布局
- Objective-C内存布局
- C的内存布局
- Objective-C内存布局
- Objective-C内存布局
- c/c++ 内存布局
- spark streaming 应用程序监控
- 32位与64位系统基本数据类型的字节数
- 【MySQL】查询解析器
- 3月31日,TableDemoMVC,每日20行。
- hdu 4865 项目管理(STL)
- c++内存布局最完整
- java_入门基础学习(二)
- 安装OracleXE
- 单例遇上多线程
- CMake和Make之间的区别-学习
- 集
- hdu 水果(map)
- Ubuntu16.04下将Python3设置为默认
- 基本git命令