C++对象的数据成员.docxVIP

  1. 1、本文档共9页,可阅读全部内容。
  2. 2、原创力文档(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。
  3. 3、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载
  4. 4、如文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“版权申诉”(推荐),也可以打举报电话:400-050-0827(电话支持时间:9:00-18:30)。
  5. 5、该文档为VIP文档,如果想要下载,成为VIP会员后,下载免费。
  6. 6、成为VIP后,下载本文档将扣除1次下载权益。下载后,不支持退款、换文档。如有疑问请联系我们
  7. 7、成为VIP后,您将拥有八大权益,权益包括:VIP文档下载权益、阅读免打扰、文档格式转换、高级专利检索、专属身份标志、高级客服、多端互通、版权登记。
  8. 8、VIP文档为合作方或网友上传,每下载1次, 网站将根据用户上传文档的质量评分、类型等,对文档贡献者给予高额补贴、流量扶持。如果你也想贡献VIP文档。上传文档
查看更多
C++对象的数据成员 C++对象的数据成员 数据成员的布局 对于一个类来说它的对象中只存放非静态的数据成员,但是除此之外,编译器为了实现virtual功能还会合成一些其它成员插入到对象中。我们来看看这些成员的布局。 C++ 标准的规定 在同一个Access Section(也就是private,public,protected片段)中, 要求较晚出现的数据成员处在较大的内存中。这意味着同一个片段中的数据成员并不需要紧密相连,编译器所做的成员对齐就是一个例子。 允许编译器将多个Acess Section的顺序自由排列,而不必在乎它们的声明 次序。但似乎没有编译器这样做。 对于继承类,C++标准并未指定是其基类成员在前还是自己的成员在前。 对于虚基类成员也是同样的未予规定。 一般的编译器怎么做? 看一小段代码: class X{ public: int a; virtual void vfc(){}; }; int main() { using namespace std; X x; coutx.a xendl; system(pause); } 在VS2010和VC6.0中运行的结果都是地址值x.a比x大4,可见说vc的vptr放在对象的最前面此言非虚。 对于Vbptr来说,有好几种方法,在这儿我们只看看VC的实现原理: 对于由虚拟继承而得的类,VC会在其每一个对象中插入一个Vbptr,这个Vbptr指向vitual base class table(我称之为虚基类表)。虚基类表中则存放有其虚基类子对象相对于虚基类指针的偏移量。例如声明如class Y:virtual public X的类的virtual base class table的虚基类表中当存储有X对象相对于Vbptr的偏移量。 对象成员或基类对象成员后面的填充空白不能为其它成员所用 看一段代码: class X{ public: int x; char c; }; class X2:public X { public:char c2; }; X2的布局应当是x(4),c(1),c2(1),这么说来sizeof(X2)的值应该是8?错了,实际上是12。原因在于X后面的三个字节的填充空白不能为c2所用。也就是说X2的大小实际上为:X(8)+c2(1)+填补(3)=12。这样看来编译器似乎是那么的呆板,其实不然,看一下下面的语句会发生什么? X2 x2; X x; x2=x; 如果X后面的填充空白可以被c2使用的话,那么X2和X都将是8字节。上面的语句执行后x2.c2的值会是多少?一个不确定的值!这样的结果肯定不是我们想要的。 Vptr与Vbptr 1 在多继承情况下,即使是多虚拟继承,继承而得的类只需维护一个Vbptr; 而多继承情况下Vptr则可能有要维护多个Vptr,视其基类有几个有虚函数。 一条继承线路只有一个Vptr,但可能有多个Vbptr,视有几次虚拟 继承而定。换言之,对于一个继承类对象来说,不需要新合成vptr,而是使用其基类子对象的vptr。而对于一个虚拟继承类来说,必须新合成一个自己的Vbptr。 如: class X{ virtual void vf(){}; }; class X2:virtual public X { virtual void vf(){}; }; class X3:virtual public X2 { virtual void vf(){}; } X3将包含有一个Vptr,两个Vbptr。确切的说这两个Vbptr一个属于X3,一个属于X3的子对象X2,X3通过其Vbptr找到子对象X2,而X2通过其Vbptr找到X。 其中差别在于vptr通过一个虚函数表可以确切地知道要调用的函数,而Vbptr通过虚基类表只能够知道其虚基类子对象的偏移量。这两条规则是由虚函数与虚拟继承的实现方式,以及受它们的存取方式和复制控制的要求决定的。 数据成员的存取 静态数据成员相当于一个仅对该类可见的全局变量,因为程序中只存在一个静态数据成员的实例,所以其地址在编译时就已经被决定。不论如何静态数据成员的存取不会带来任何额外负担。 非静态数据成员的存取,相当于对象起始地址加上偏移量。效率上与C struct成员的效率等同。因为它的偏移量在编译阶段已经确定。但有一种情况例外:pt-x=0.0。当通过指针或引用来存取——x而x又是虚基类的成员的时候。因为必须要等到执行期才能知道pt指向的确切类型,所以必须通过一个间接导引才能完成。 小结 在VC中数据成员的布局顺序为: vptr部分(如果基类有,则继承基类的) vbptr (如果需要) 基类成员(按声明顺序) 自身数

您可能关注的文档

文档评论(0)

158****9567 + 关注
实名认证
文档贡献者

该用户很懒,什么也没介绍

1亿VIP精品文档

相关文档