• 关于类对象大小的 sizeof 计算问题


    之前看了很多关于类对象大小问题的计算,今天闲着没事就来整理整理,写写自己的看法。

    首先,来看看一个只有构造函数和析构函数的空类:

    1. #include <iostream>  
    2. using namespace std;  
    3. class Base  
    4. {  
    5. public:  
    6.     Base();  
    7.     ~Base();  
    8. };  
    9. int main(int argc, char *argv[])  
    10. {  
    11.     cout << sizeof(Base) << endl;  
    12. }  

    输出结果为:1

          因为一个空类也要实例化,所谓类的实例化就是在内存中分配一块地址,每个实例在内存中都有独一无二的地址。同样空类也会被实例化,所以编译器会给空类隐含的添加一个字节,这样空类实例化之后就有了独一无二的地址了。所以空类的sizeof为1。 而析构函数,跟构造函数这些成员函数,是跟sizeof无关的,也不难理解因为我们的sizeof是针对实例,而普通成员函数,是针对类体的,一个类的成员函数,多个实例也共用相同的函数指针,所以自然不能归为实例的大小。

         如果给这个类添加成员变量,最后输出的大小就是这些成员变量的大小之和(这里涉及到一个成员对齐问题,不再叙述了)。

    接下来再来看一个有继承的例子:

    1. #include <iostream>  
    2. using namespace std;  
    3. class Base  
    4. {  
    5. public:  
    6.     Base();                  
    7.     ~Base();           
    8.     void set_num(int num)    //普通成员函数  
    9.     {  
    10.         a=num;  
    11.     }  
    12. private:  
    13.     int  a;                  //占4字节  
    14.     char *p;                 //4字节指针  
    15. };  
    16. class Derive:public Base  
    17. {  
    18. public:  
    19.     Derive():Base(){};       
    20.     ~Derive(){};  
    21. private:  
    22.     static int st;         //非实例独占  
    23.         int  d;                //占4字节  
    24. };  
    25. int main(int argc, char *argv[])   
    26. {   
    27.     cout<<sizeof(Base)<<endl;  
    28.     cout<<sizeof(Derive)<<endl;  
    29.     return 0;  
    30. }  

    输出结果为:8   12

    结果很显然, Base 类按4字节对齐,所以是8个字节,Derive 类中不但继承了Base 类的两个成员变量,还多了两个成员变量,但大小却只有12字节,可以得出:静态变量在计算时是不做考虑的。

    上面的例子中都没有涉及到虚函数,下面看个有虚函数的例子:

    1. #include <iostream>  
    2. using namespace std;  
    3. class Base  
    4. {  
    5. public:  
    6.     Base() {}  
    7.     virtual ~Base() {}  
    8. };  
    9. int main(int argc, char *argv[])  
    10. {  
    11.     cout << sizeof(Base) << endl;  
    12.     return 0;  
    13. }  

    输出结果为:4

          和第一个程序相比,这个类中,析构函数变成了虚函数,类的大小也变成了4字节,这是因为有了虚函数,编译器就会为类创建一个虚函数表(vtable),并创建一个指针(vptr)指向这个虚函数表。所以类大小变为4字节。如果在 Base 类中再添加新的虚函数,该类的大小还是不会变,因为指向虚函数的指针是放在虚函数表中的,指向虚函数表的指针不会变。

    如果在这个类中添加数据成员,就会在4字节的基础上对象大小。

    下面再来看看虚函数的继承问题:

    1. #include <iostream>  
    2. using namespace std;  
    3. class Base  
    4. {  
    5. public:  
    6.     Base();                  
    7.     virtual ~Base();           
    8.     void set_num(int num)    //普通成员函数  
    9.     {  
    10.         a=num;  
    11.     }  
    12. private:  
    13.     int  a;                  //占4字节  
    14.     char *p;                 //4字节指针  
    15. };  
    16. class Derive:public Base  
    17. {  
    18. public:  
    19.     Derive():Base(){};       
    20.     ~Derive(){};  
    21.     virtual void foo() { }  
    22. private:  
    23.     static int st;         //非实例独占  
    24.         int  d;                //占4字节  
    25. };  
    26. int main(int argc, char *argv[])   
    27. {   
    28.     cout<<sizeof(Base)<<endl;  
    29.     cout<<sizeof(Derive)<<endl;  
    30.     return 0;  
    31. }  

    输出结果为:12    16

          Base类的大小为12字节很显然,Derive 类中,虽然有一个虚函数 foo ,但是因为它是从Base 类继承的,所以也继承了其虚函数表,并没有创新新的虚函数表,只是在继承下来的表中添加了一项,所以大小为16字节。

    再来看看一个虚继承的例子:

    1. #include <iostream>  
    2. using namespace std;  
    3. class Base  
    4. {  
    5. public:  
    6.     Base();                  
    7.     virtual ~Base();           
    8.     void set_num(int num)    //普通成员函数  
    9.     {  
    10.         a=num;  
    11.     }  
    12. private:  
    13.     int  a;                  //占4字节  
    14.     char *p;                 //4字节指针  
    15. };  
    16. class Derive:virtual public Base  
    17. {  
    18. public:  
    19.     Derive():Base(){};       
    20.     ~Derive(){};  
    21.     virtual void foo() { }  
    22. private:  
    23.     static int st;         //非实例独占  
    24.         int  d;                //占4字节  
    25. };  
    26. int main(int argc, char *argv[])   
    27. {   
    28.     cout<<sizeof(Base)<<endl;  
    29.     cout<<sizeof(Derive)<<endl;  
    30.     return 0;  
    31. }  

    输出结果为:12    20

          这里由于虚继承而引入了一个间接的指针(vbc),该指针是指向虚函数表的一个slot,表中存放着该slot中存放虚基类子对象的偏移量的负值。所以大小比之前多了4字节。就算同时虚继承自两个类,也只会有一个这样的间接指针,也就是大小也只多4字节。

    说到这里,关于类对象大小问题的计算应该差不多了。

  • 相关阅读:
    2021-04-14:判断二叉树是否是满二叉树?
    2021-04-13:判断二叉树是否是平衡二叉树?
    2021-04-12:判断二叉树是否是搜索二叉树?
    2021-04-11:判断二叉树是否是完全二叉树?
    2021-04-10:给定两个可能有环也可能无环的单链表,头节点head1和head2。请实现一个函数,如果两个链表相交,请返回相交的 第一个节点。如果不相交,返回null。【要求】如果两个链表长度之和为N,时间复杂度请达到O(N),额外空间复杂度 请达到O(1)。
    2021-04-09:rand指针是单链表节点结构中新增的指针,rand可能指向链表中的任意一个节点,也可能指向null。给定一个由Node节点类型组成的无环单链表的头节点 head,请实现一个函数完成这个链表的复制,并返回复制的新链表的头节点。 【要求】时间复杂度O(N),额外空间复杂度O(1) 。
    2021-04-08:给定一个单链表的头节点head,请判断该链表是否为回文结构。
    影响成败的小细节
    一个完整IC项目需要走过的流程
    linux 下shell中if的“-e,-d,-f”是什么意思 ------ 转载
  • 原文地址:https://www.cnblogs.com/iloveyoucc/p/2409084.html
Copyright © 2020-2023  润新知