- 1、原创力文档(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。。
- 2、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载。
- 3、如文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“版权申诉”(推荐),也可以打举报电话:400-050-0827(电话支持时间:9:00-18:30)。
- 4、该文档为VIP文档,如果想要下载,成为VIP会员后,下载免费。
- 5、成为VIP后,下载本文档将扣除1次下载权益。下载后,不支持退款、换文档。如有疑问请联系我们。
- 6、成为VIP后,您将拥有八大权益,权益包括:VIP文档下载权益、阅读免打扰、文档格式转换、高级专利检索、专属身份标志、高级客服、多端互通、版权登记。
- 7、VIP文档为合作方或网友上传,每下载1次, 网站将根据用户上传文档的质量评分、类型等,对文档贡献者给予高额补贴、流量扶持。如果你也想贡献VIP文档。上传文档
查看更多
C+啖对象内存结构
首先介绍一下C++中有继承关系的类对象内存的布局:在 C++中,如果类中 有虚函数,那么它就会有一个虚函数表的指针 __vfptr,在类对象最开始的内存 数据中。之后是类中的成员变量的内存数据。
对于子类,最开始的内存数据记录着父类对象的拷贝 (包括父类虚函数表指 针和成员变量)。 之后是子类自己的成员变量数据。
对于子类的子类,也是同样的原理。但是无论继承了多少个子类,对象中始 终只有一个虚函数表指针。
Base
Child
Child
为了探讨C+啖对象的内存布局,先来写几个类和函数
首先写一个基类:
class Base
{
public:
virtual void f() { cout Base::f en dl; } virtual void g() { cout Base::g en dl; } virtual void h() { cout Base::h en dl; } int base;
protected:
private:
然后,我们多种不同的继承情况来研究子类的内存对象结构
无虚函数集继承
//子类1,无虚函数重载
class Childl : public Base
{
public:
virtual void f1() { cout Child1::f1 endl; } virtual void g1() { cout Child1::g1 en dl; } virtual void h1() { cout Child1::h1 en dl; } int child1;
protected:
private:
};
这个子类Child1没有继承任何一个基类的虚函数,因此它的虚函数表如下 图:
Childl
Childl ::glQ
Childl 1()
我们可以看出,子类的虚函数表中,先存放基类的虚函数,在存放子类自己 的虚函数。
有一个虚函数继承
当子类重载了父类的虚函数,则编译器会将子类虚函数表中对应的父类的虚 函数替换成子类的函数。
全部虚函数都继承
//子类3,全部虚函数重载
class Child3 : public Base
{
public:
virtual void f() { cout Child3::f en dl; } virtual void g() { cout Child3::g en dl; } virtual void h() { cout Child3::h en dl; } protected:
int x;
private:
};
Ease
int base,vfptrint base;int child 1;
int base,
vfptr
int base;
int child 1;
多重继承
多重继承,即类有多个父类,这种情况下的子类的内存结构和单一继承有所 不同。
我们可以看到,当子类继承了多个父类,那么子类的内存结构是这样的: 类的内存中,顺序
5.菱形继承
Ease-l
讪
ini兀曰:
■
Pl -1门】
匚1皿
infl j(=:L:
njl =: vqotr
f 3SCL::flj
ml =::[;
罢孙:爼4(」
im 3出?1;
i:xW5;rv)
ml i口]-
6.单一虚拟继
虚拟继承的子类的内存结构,和普通继承完全不同。虚拟继承的子类,有单 独的虚函数表,另外也单独保存一份父类的虚函数表,两部分之间用一个四个 字节的0作为分界。子类的内存中,首先是自己的虚函数表,然后 是子类的数据成员,然后是0x0,之后就是父类的虚函数表,之后是父类的数 据成员。
如果子类没有自己的虚函数,那么子类就不会有虚函数表,但是子类数据和 父类数据之间,还是需要0x0来间隔。
因此,在虚拟继承中,子类和父类的数据,是完全间隔的,先存放子类自己 的虚函数表和数据,中间以0x分界,最后保存父类的虚函数和数据。如果子类 重载了父类的虚函数,那么则将子类内存中父类虚函数表的相应函数替换。
7.菱形虚拟继承
Egg
if . i ■!virlmil.nl I i :irl
if . i ■!
virlmil.
nl I i :irl
hi tearcS-
也3
irii. i
肥IH气(°
如诜■<;
s-s上珥:
Chilili*
请?T聃-?!_
A
汹1:
Em*?:」:“
iiir: W
%罠足?ljry
Ddl uiulcS
0
vfilfe8
-v
-ii I
Kw.L J:C;
二INIf,呵卩
CS1J5:出;:
结论:
(1) 对于基类,如果有虚函数,那么先存放虚函数表指针,然后存放自己 的数据成员;如果没有虚函数,那么直接存放数据成员。
(2) 对于单一继承的类对象,
原创力文档


文档评论(0)