网上关于c++对象布局的文章挺多,而且《深度探索c++对象模型》(Inside The C++ Object Model 侯捷 译)一书中也很详细地介绍。如果你一点都不了解C++对象的布局,我推荐你看看《深度探索c++对象模型》的第三章,如果你意犹未尽下面的两个系列都很不错:

一是陈皓的《C++ 对象的内存布局》图文并貌,写得很是详细。地址是http://blog.csdn.net/haoel/archive/2008/10/15/3081328.aspx。

二是玄机逸士的《对象内存布局》系列则几乎把每种可能性列出来了,尤为详尽。地址是http://blog.csdn.net/pathuang68/archive/2009/04/23/4101970.aspx。

读了以上好文,对于c++对象的布局其实应该是山水了然于胸了,不过我最在写一个用c++模仿C#事件机制的东东,但发觉得有很几处细节不是很明了,而且陈皓朋友也在最后提出了一个问题,虽然有网友答之,仍语言不详。而我却必需深入了解此,才能真正实做出C++的事件。

此文应该算是狗尾续貂之作,高手可能会不屑一顾。不过希望还是对一些朋友有帮助。因为对于普通类的对象布局,前人备述已,我也不太可能写出什么新意来,更多的是因为我比较懒。呵呵,本文所探讨的就是那种极端复杂的菱形结构如下:

class A ;

class B : virtual public A;

class C : virtual public B;

class D : public B,public C;

好了,来了,我们从最基础的的讨论起。当c++支持virtual base class 时,就会多了一些额外负担,当class中内含一个或多个virtual base class subobject时,将分成两个部分,一个不变局部和一个共享局部。最初的方案是为每一个虚基类安插一个指针指向这个虚基类,其缺点是为了负担太重,而且当虚继承链加长时,导致间接存取时间加长(需通过多次跳转)。因此有两种解决方案(《深入》一书中所提):

一、是引入virtual base class table,不管多少个虚基类,总是只有一个指针指向它,这个virtual base class table(VBTBL)包括真正的 virtual base class 指针。

二、Bjarne的办法是在virtual function table中放置virtual base class的offset,而非地址,这个offset在virtual function table 的负位置(正值是索引virtual function,而负值则方向盘引到virtual base class offsets)。

我用vc2003观测到的实际情况是。在类中增加一个指针(VBPTR)指向一个VBTBL,这个VBTBL的第一项记载的是从VBPTR 与本类的偏移地址,如果本类有虚函数,那么第一项是FF FF FF FC(也就是-4),如果没有则是零,第二项起是VBPTR与本类的虚基类的偏移值。vc2003的这种方案个人觉得没有Bjarne的好,一是要多一个指针,二是因为VBPTR与虚函数表分开设计,也不便于修改。至于其它编译器,因为我跟其它编译器不熟,所以也就没有实测它们。

下面给出对于类定义

struct B1

{

int a;

int b;

};

struct B2

{

virtual void foo(void);

int c;

int d;

};

struct Test : virtual public B1, virtual public B2

{

virtual void func1(void);

virtual void func2(void);

virtual void func3(void);

int X;

};

一个Test 对象的内存布局图,我们可以清楚的看到在VS2003中VBPTR以及VBTBL的结构以及其相关的内容是什么意义。以及Bjarne的方案的优点。

最后我们来看一个完整的例子以及内存结构布局。图后有相关代码。

代码如下:

struct A

{

A(int v=100):X(v){};

virtual void foo(void){}

int X;

};

struct B :virtual public A

{

B(int v=10):Y(v),A(100){};

virtual void fooB(void){}

int Y;

};

struct C : virtual public A

{

C(int v=20):Z(v),A(100){}

virtual void fooC(void){}

int Z;

};

struct D : public B, public C

{

D(int v =40):B(10),C(20),A(100),L(v){}

virtual void fooD(void){}

int L;

};

int _tmain(int argc, _TCHAR* argv[])

{

A a;

int *ptr;

ptr = (int*)&a;

cout << ptr << " sizeof = " << sizeof(a) <<endl;

for(int i=0;i<sizeof(A)/sizeof(int);i++)

{

if(ptr[i] < 10000)

{

cout << dec << ptr[i]<<endl;

}

else cout << hex << ptr[i] <<" = " << hex << * ((int*)(ptr[i])) <<endl;

}

cout << "--------------------------------------" <<endl;

B b;

ptr = (int*)&b;

cout <<"addr:" << ptr << " sizeof = " << sizeof(b) <<endl;

for(int i=0;i<sizeof(B)/sizeof(int);i++)

{

if(ptr[i] < 10000)

{

cout << dec << ptr[i]<<endl;

}

else cout << hex << ptr[i] <<" = " << hex << * ((int*)(ptr[i])) <<endl;

}

cout << "--------------------------------------" <<endl;

D d;

ptr = (int*)&d;

cout <<"addr:" << ptr << " sizeof = " << sizeof(d) <<endl;

for(int i=0;i<sizeof(D)/sizeof(int);i++)

{

if(ptr[i] < 10000)

{

cout << dec << ptr[i]<<endl;

}

else cout << hex << ptr[i] <<" = " << hex << * ((int*)(ptr[i])) <<endl;

}

return 0;

}

C++虚继承(十) --- 谈谈陈皓遗留的问题相关推荐

  1. 基本语言细节---C++ 虚函数表解析 陈皓

    C++ 虚函数表解析 陈皓 http://blog.csdn.net/haoel 前言 C++中的虚函数的作用主要是实现了多态的机制.关于多态,简而言之就是用父类型别的指针指向其子类的实例,然后通过父 ...

  2. C++虚继承(三) --- C++ 对象的内存布局(下)(陈皓)

    C++ 对象的内存布局(下) 陈皓 http://blog.csdn.net/haoel <<<点击这里查看上篇 重复继承 下面我们再来看看,发生重复继承的情况.所谓重复继承,也就是 ...

  3. C++虚继承(二) --- C++ 对象的内存布局(上)(陈皓)

    C++ 对象的内存布局(上) 陈皓 http://blog.csdn.net/haoel 点击这里查看下篇>>> 前言 07年12月,我写了一篇<C++虚函数表解析>的文 ...

  4. 陈皓:谈谈职业规划——CSDN对我的采访

    转载自:http://blog.csdn.net/haoel/article/details/1688104 职业规划就像软件工程 电信.银行等行业一直是许多人非常向往的工作单位,清差厚禄,旱涝保收, ...

  5. 陈皓:谈谈数据安全和云存储

    本文讲的是 : 陈皓:谈谈数据安全和云存储   ,  前些天,创新工场李开复同学在2012博鳌亚洲论坛表示: "你们有多少人丢过手机?大概有15%.你们有多少人数据放在微软掉过的?我想不见得 ...

  6. 图灵访谈之三十二:我的精神家园——陈皓专访

    图灵访谈之三十二:我的精神家园--陈皓(@左耳朵耗子)专访 推荐34收藏 芝兰生于深谷,不以无人而不芳 .君子修身养德,不以穷困而改志. "码农人物志"第二期码农代表:陈皓(@左耳 ...

  7. 图灵访谈之三十二:我的精神家园——陈皓(@左耳朵耗子)专访

    芝兰生于深谷,不以无人而不芳 .君子修身养德,不以穷困而改志. "码农人物志"第二期码农代表:陈皓(@左耳朵耗子),酷壳coolshell.cn博主. 14年软件开发相关工作经验, ...

  8. 程序员技术练级攻略--原作者:陈皓

    复制过来,作参考用. 程序员技术练级攻略 2011年07月18日  陈皓 评论 596 条评论  754,044 人阅读 月光博客6月12日发表了<写给新手程序员的一封信>,翻译自< ...

  9. [转载] 陈皓——程序员技术练级攻略

    PS:原文出自酷壳上的陈皓对程序员从入门到精通的攻略,让你感受一下真正的大神吧!又是阿里人,他的文章真心不错,希望对你也有用.原文地址:http://coolshell.cn/articles/499 ...

最新文章

  1. RDKit | 基于RDKit绘制黑白颜色的分子
  2. Android AsyncTask简单用法
  3. 网络测试及故障诊断方法及工具
  4. linux awk列数据处理工具使用示例
  5. 闪回的用途与实战(闪回表,闪回删除,闪回重名删除,闪回版本查询)
  6. 在微型计算机控制系统中常用的报警方式中,微机控制技术复习题
  7. static在内存层面的作用_C++内存管理笔记
  8. docker 获取镜像
  9. p语言是python吗-为什么说Python是一门动态语言--Python的魅力
  10. BZOJ4892:[TJOI2017]dna(hash)
  11. Android简单的发短信示例
  12. 使用AForge录制视频
  13. Wireshark 用户使用手册 ———— 配置与属性
  14. 谈谈企业的持续交付流水线设计
  15. 一坑未平一坑又起——圆锥曲线1-1 椭圆的定义中的东西
  16. ssm项目整合与功能开发(注解开发)
  17. 2006-2010年美国纽约市空气中一氧化碳污染情况分析
  18. C++实现勒让德多项式(附完整源码)
  19. 读俞敏洪老师自传有感
  20. 如何安装博通 BCM43142 网卡驱动

热门文章

  1. Hive和Hadoop及RDBMS关系
  2. jwt:介绍以及创建token
  3. 工程搭建:搭建子工程之分布式id生成器
  4. 类属性-类属性的定义及使用
  5. 设计模式之_动态代理_05
  6. mysql5.58_mysql5.58编译安装手记
  7. fedora yum mysql_Fedora14使用yum安装mysql
  8. c# wpf 面试_WPF 基础面试题及答案(一)
  9. python2与python3共存_【python】--python2与python3 共存
  10. data中的数据如何在innerhtml中调用_Vuex中调用state数据