首先重新回顾一下关于类/对象大小的计算原则:

类大小计算遵循结构体对齐原则

第一个数据成员放在offset为0的位置

其它成员对齐至min(sizeof(member),#pragma pack(n)所指定的值)的整数倍。

整个结构体也要对齐,结构体总大小对齐至各个成员中最大对齐数的整数倍。

win32 可选的有1, 2, 4, 8, 16
linux 32 可选的有1, 2, 4

类的大小与数据成员有关与成员函数无关
类的大小与静态数据成员无关
虚继承对类的大小的影响
虚函数对类的大小的影响

下面通过实例来展示虚继承和虚函数对类大小造成的影响。

测试环境为:Win32 + Vs2008

一、只出现虚继承的情况

 C++ Code 
1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

 
#include <iostream>


using 
namespace std;

class BB

{


public :

      
int bb_ ;

};

class B1 : 
virtual 
public BB

{


public :

      
int b1_ ;

};

class B2 : 
virtual 
public BB

{


public :

      
int b2_ ;

};

class DD : 
public B1, 
public B2

{


public :

      
int dd_ ;

};

int main (
void)

{

      cout<<
sizeof (BB)<< endl;

      cout<<
sizeof (B1)<< endl;

      cout<<
sizeof (DD)<< endl;

B1 b1 ;

      
int** p ;

cout<<&b1 <<endl;

      cout<<&b1 .bb_<< endl;

      cout<<&b1 .b1_<< endl;

p = (
int **)&b1;

      cout<<p [
][
]<<endl;

      cout<<p [
][
]<<endl;

DD dd ;

      cout<<&dd <<endl;

      cout<<&dd .bb_<< endl;

      cout<<&dd .b1_<< endl;

      cout<<&dd .b2_<< endl;

      cout<<&dd .dd_<< endl;

      p = (
int **)&dd;

      cout<<p [
][
]<<endl;

      cout<<p [
][
]<<endl;

      cout<<endl ;

      cout<<p [
][
]<<endl;

      cout<<p [
][
]<<endl;

BB* pp ;

pp = &dd ;

      dd.bb_ = 

//对象的内存模型在编译时就已经确定了,否则无法定义类的对象,因为要开辟内存
      
int base = pp-> bb_;     
// 通过间接访问 (其实pp 已经偏移了20 ),这需要运行时的支持
      cout<<
"dd.bb_=" <<base<< endl;

return 
;

}

从输出的地址和虚基类表成员数据可以画出对象内存模型图:

virtual base table

本类地址与虚基类表指针地址的差

虚基类地址与虚基类表指针地址的差

virtual base table pointer(vbptr)

从程序可以看出pp是BB* 指针,pp首先指向dd内存,当执行pp->bb_时,先找到首个vbptr,找到虚基类BB地址与虚基类表指针地址的差,也即是20,接着pp偏移20个字节指向了dd对象中的BB部分,然后就访问到了bb_,这是在运行时才做的转换。

二、只出现虚函数的情况

 C++ Code 
1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

 
#include <iostream>


using 
namespace std;

class Base

{


public :

    
virtual 
void Fun1()

    {

        cout << 
"Base::Fun1 ..." << endl;

    }

virtual 
void Fun2()

    {

        cout << 
"Base::Fun2 ..." << endl;

    }

    
int data1_ ;

};

class Derived : 
public Base

{


public :

    
void Fun2 ()

    {

        cout << 
"Derived::Fun2 ..." << endl;

    }

    
virtual 
void Fun3()

    {

        cout << 
"Derived::Fun3 ..." << endl;

    }

    
int data2_ ;

};

typedef 
void (* FUNC)(
void );

int main (
void)

{

    cout << 
sizeof (Base) << endl;

    cout << 
sizeof (Derived) << endl;

    Base b ;

    
int **p = (
int **)& b;

    FUNC fun = (FUNC) p[
][
];

    fun();

    fun = (FUNC )p[
][
];

    fun();

    cout << endl ;

Derived d ;

    p = (
int **)&d;

    fun = (FUNC )p[
][
];

    fun();

    fun = (FUNC )p[
][
];

    fun();

    fun = (FUNC )p[
][
];

    fun();

return 
;

}

从输出的函数体可以画出对象内存模型图:

vtbl:虚函数表(存放虚函数的函数指针)

vptr:虚函数表指针

从输出可以看出,Derived类继承了Base::Fun1,而覆盖了Fun2,此外还有自己的Fun3。注意,因为Fun3是虚函数,才会出现在虚函数表,如果是一般函数是不会的,因为不用通过vptr间接访问。

三、虚继承与虚函数同时出现的情况:

 C++ Code 
1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

70

71

72

73

74

75

76

77

78

79

80

81

82

83

84

85

86

87

88

89

90

91

92

93

94

95

96

97

98

99

100

101

102

103

104

 
#include <iostream>


using 
namespace std;

class BB

{


public :

      
virtual 
void vfbb()

     {

           cout<<
"BB::vfbb" <<endl;

     }

      
virtual 
void vfbb2()

     {

           cout<<
"BB::vfbb2" <<endl;

     }

      
int bb_ ;

};

class B1 : 
virtual 
public BB

{


public :

      
virtual 
void vfb1()

     {

           cout<<
"B1::vfb1" <<endl;

     }

      
int b1_ ;

};

class B2 : 
virtual 
public BB

{


public :

      
virtual 
void vfb2()

     {

           cout<<
"B2::vfb2" <<endl;

     }

      
int b2_ ;

};

class DD : 
public B1, 
public B2

{


public :

      
virtual 
void vfdd()

     {

           cout<<
"DD::vfdd" <<endl;

     }

      
int dd_ ;

};

typedef 
void (* FUNC)(
void);

int main (
void)

{

      cout<<
sizeof (BB)<< endl;

      cout<<
sizeof (B1)<< endl;

      cout<<
sizeof (DD)<< endl;

BB bb ;

      
int** p ;

      p = (
int **)&bb;

      FUNC fun ;

      fun = (FUNC )p[
][
];

      fun();

      fun = (FUNC )p[
][
];

      fun();

      cout<<endl ;

B1 b1 ;

     

      p = (
int **)&b1;

      fun = (FUNC )p[
][
];

      fun();

      fun = (FUNC )p[
][
];

      fun();

      fun = (FUNC )p[
][
];

      fun();

cout<<p [
][
]<<endl;

      cout<<p [
][
]<<endl;

      cout<<endl ;

DD dd ;

      p = (
int **)&dd;

      fun = (FUNC )p[
][
];

      fun();

      fun = (FUNC )p[
][
]; 
// DD::vfdd 挂在 B1::vfb1的下面
      fun();

      fun = (FUNC )p[
][
];

      fun();

      fun = (FUNC )p[
][
];

      fun();

      fun = (FUNC )p[
][
];

      fun();

     

      cout<<p [
][
]<<endl;

      cout<<p [
][
]<<endl;

      cout<<p [
][
]<<endl;

      cout<<p [
][
]<<endl;

return 
;

}

从输出的虚基类表成员数据和虚函数体可以画出对象内存模型图:

注意:如果没有虚继承,则虚函数表会合并,一个类只会存在一个虚函数表和一个虚函数表指针(同个类的对象共享),当然也不会有
虚基类表和虚基类表指针的存在。

参考:

C++ primer 第四版
Effective C++ 3rd
C++编程规范

从零开始学C++之虚继承和虚函数对C++对象内存模型造成的影响的更多相关文章

  1. c++虚继承与虚函数

    学习继承与多态时看到这两个概念,记录整理. 虚继承与虚函数都是用virtual关键字实现,虚继承为了防止多重继承,而虚函数为了实现多态. 是几个例子. 虚继承: class A{}; class B: ...

  2. C++多重继承分析——《虚继承实现原理(虚继承和虚函数)》

    博客转载:https://blog.csdn.net/longlovefilm/article/details/80558879 一.虚继承和虚函数概念区分 虚继承和虚函数是完全无相关的两个概念. 虚 ...

  3. C++对象内存模型2 (虚函数,虚指针,虚函数表)

    从例子入手,考察如下带有虚函数的类的对象内存模型: class A { public: virtual void vfunc1(); virtual void vfunc2(); void func1 ...

  4. C++虚函数、虚继承、对象内存模型(转)

    参考:http://blog.csdn.net/hxz_qlh/article/details/14633361 需要注意的是虚继承.多重继承时类的大小.

  5. C++基础 (6) 第六天 继承 虚函数 虚继承 多态 虚函数

    继承是一种耦合度很强的关系 和父类代码很多都重复的 2 继承的概念 3 继承的概念和推演 语法: class 派生类:访问修饰符 基类 代码: … … 4 继承方式与访问控制权限 相对的说法: 爹派生 ...

  6. C++对象内存模型2 (虚函数,虚指针,虚函数表)(转)

    class A { public: virtual void vfunc1(); virtual void vfunc2(); void func1(); void func2(); virtual ...

  7. 从零开始学 Web 之 JavaScript(三)函数

    大家好,这里是「 Daotin的梦呓 」从零开始学 Web 系列教程.此文首发于「 Daotin的梦呓 」公众号,欢迎大家订阅关注.在这里我会从 Web 前端零基础开始,一步步学习 Web 相关的知识 ...

  8. C++学习笔记----4.5 C++继承时的对象内存模型

    推荐阅读:http://blog.csdn.net/randyjiawenjie/article/details/6693337 最近研究了一下,C++继承的内存对象模型.主要是读了读http://b ...

  9. 记录:C++类内存分布(虚继承与虚函数)

    工具:VS2013 先说一下VS环境下查看类内存分布的方法: 先选择左侧的C/C++->命令行,然后在其他选项这里写上/d1 reportAllClassLayout,它可以看到所有相关类的内存 ...

随机推荐

  1. JSR303 Bean Validation 技术规范特性概述

    概述 Bean Validation 规范 Bean 是 Java Bean 的缩写,在 Java 分层架构的实际应用中,从表示层到持久化层,每一层都需要对 Java Bean 进行业务符合性验证,如 ...

  2. bat启动/停止oracle服务

    原文:bat启动/停止oracle服务 自己的电脑比较慢,尤其装了oracle10g后,服务开启和关闭用bat文件操作省事点 开启服务 @echo offnet start OracleService ...

  3. Erlang常用代码段

    十六进制字符串转为二进制 hex_to_bin(Bin) -> hex2bin(Bin). hex2bin(Bin) when is_binary(Bin) -> hex2bin(bina ...

  4. 学习使用简单的php

    配置文件在:/etc/php5/$中,不同的模式含有自己的php.ini配置文件.php可以运行于多种模式:cgi.fastcgi.cli.web模块模式等4种: 我现在使用的模式是cli模式,这里进 ...

  5. ReSharper 8.1支持TypeScript语言之代码检查特征

    自ReSharper 8.1发布以来,就支持TypeScript.其在TypeScript语言拼写帮助和代码完成中,几乎是一个里程碑的发展,这是令人激动的改进. 支持TypeScript效果就目前测试 ...

  6. 安装Visual Source Safe 2005 - 初学者系列 - 学习者系列文章

    本文介绍微软的文档管理工具Visual Source Safe 2005的安装 从下列地址获取该工具: ed2k://|file|en_vss_2005.iso|108048384|C4BEC1EC3 ...

  7. gradle测试与线上打包

    首先,第一反应理所当然的是profile : <?xml version="1.0" encoding="UTF-8"?> <beans xm ...

  8. java中实现与.net的format格式化字符串输出

    Java中的格式化字符串 System.out.println(MessageFormat.format("name={0}", "张三")); .net中的格 ...

  9. idea执行go

    因为经常在不同的地方调代码,每次都调整环境很麻烦,于是在犯懒的时候发现了更直接简便的办法,关于idea集成go环境的,不需要按部就班的部署. 首先下代码,比如https://github.com/sa ...

  10. java UDP网路编程

    大家都知道java中的socket网络编程,而其采用的协议分别有tcp和udp协议两种. 通常的理解tcp协议类似于打电话,udp类似于发短信.前者是线程安全的,但是效率比较低.后者则刚好相反. 今天 ...