多态的类,内存布局是怎样的

关于类的内存布局主要是考某个类所占的内存大小,通过以下几个案例加以分析

  1. 虚继承:如果是虚继承,那么就会为这个类创建一个虚表指针,占用4个字节

     #include<iostream>
     class A{
             public:
             int a;
     };//sizeof(A) = 4,因为a是整形,占用4个字节
    
     class B:virtual piblic A{
         public:
             int b;
     };//sizeof(B) = 4(A副本) + 4(虚表指针占用4字节) +4(变量b占4个字节) = 12
    
     class C:virtual public B{
     };//sizeof(c) = 12(B的副本) + 4(虚表指针) = 16
     //如果这里改为直接继承,那么sizeof(C)为12,因为此时没有虚表指针了
  2. 多重继承:如果是以虚继承实现的多继承,记得减掉基类的副本

     #include<iostream>
     class A{
         public:
                int a;
     };//sizeof(A) = 4
    
     class B:virtual piblic A{
     };//sizeof(B) = 4+4 = 8
    
     class C:virtual public A{
     };//sizeof(C) = 4+4 = 8
    
     class D:public B, public c{
     };//sizeof(D) = 8+8-4 = 12 
     //或者 sizeof(D) 4+4+4 = 12
  3. 普通继承(含有:空类、虚函数)

     class A      //result=1  空类所占空间的大小为1  
     {     
     };    
    
     class B     //result=8  1+4   字节对齐后为 8  
     {  
         char ch;     
         virtual void func0()  {  }   
     };   
    
     class C    //result=8  1+1+4 字节对齐后为 8,没有继承的,此时类里即使出现多个虚函数,也只有一个虚指针
     {  
         char ch1;  
         char ch2;  
         virtual void func()  {  }    //也只有一个虚指针
         virtual void func1() {  }    //也只有一个虚指针
     };  
    
     class D: public A, public C  //result=12  8(C的副本)+4(整形变量    d占用4字节)=12  
     {     
         int d;     
         virtual void func()  {  }   //继承了C,C里已经有一个虚指针,此时D自己有虚函数,
         virtual void func1()  {  } //也不会创建另一个虚指针,所以D本身就变量d需要4字节
     };     
    
     class E: public B, public C  //result=20   8( B的副本)+8(C的副本)+4(E本身)=20
     {     
            int e;     
         virtual void func0()  {  }   //同理,E不会创建另一个虚指针,所以E本身就变量e需
         virtual void func1()  {  }  //要4字节
     };  
  4. 虚函数(多继承和虚函数)

class CommonBase  
{  
    int co;  
};// size = 4  

class Base1: virtual public CommonBase            
{  
public:  
    virtual void print1() {  }  
    virtual void print2() {  }  
private:  
    int b1;  
};//4(父类副本)+4(自己有虚函数,加1个虚指针空间)+4(自身变量b1)+4(虚继承再加1个虚指针空间)=16  

class Base2: virtual public CommonBase          
{  
public:  
    virtual void dump1() {  }  
    virtual void dump2() {  }  
private:  
    int b2;  
};//同理16  

class Derived: public Base1, public Base2       
{  
public:  
    void print2() {  }  
    void dump2() {  }  
private:  
    int d;  
};//16+16-4+4=32  
  • 前辈总结说:如果不是虚继承的类,即便有虚函数也不会因此增加存储空间,如果是虚继承的类,没有虚函数就添加一个虚指针空间,有虚函数不论多少个,就添加两个虚指针空间。
  • 如果此时类里有一个或多个虚函数,那么需要加1个虚指针空间,如果还是虚继承,那么需要再加1个虚指针空间,最多就2个虚指针空间。
  1. 虚继承与虚函数
class A  
{  
public:  
    virtual void aa() {  }  
    virtual void aa2() {  } //如果此时类里有一个或多个虚函数,那么需要加1个虚指针空间
private:  
    char ch[3];  
}; // 1+4 =补齐= 8   

class B: virtual public A //如果还是虚继承,那么需要再加1个虚指针空间,最多就2个虚指//针空间。
{  
public:  
    virtual void bb() {  }  
    virtual void bb2() {  }  
}; // 8(副本)+4(虚继承)+4(虚指针)= 16  
  1. 总结:如果此时类里有一个或多个虚函数,那么需要加1个虚指针空间,如果还是虚继承,那么需要再加1个虚指针空间,最多就2个虚指针空间。
全部评论

相关推荐

头像
03-03 15:53
已编辑
黑龙江大学 Java
在当前开源项目极为丰富的背景下,付费资源并不一定意味着最前沿的技术优势,在具体执行层面展示出自己的独特价值,才是简历上最重要的加分项。1.&nbsp;WebMCP&nbsp;—&nbsp;让网站主动告诉&nbsp;AI&nbsp;该怎么操作AI&nbsp;操作浏览器的方案一直靠&quot;猜&quot;——截图识别、DOM&nbsp;解析,错误率&nbsp;15-30%。WebMCP&nbsp;反过来,让网站自己声明能做什么,AI&nbsp;直接调用结构化接口,准确率接近&nbsp;100%。Chrome&nbsp;Canary&nbsp;已实装。企业内部系统的&nbsp;WebMCP&nbsp;适配目前几乎没人做,是明确的蓝海。推荐理由:简历上写的不是&quot;我会用某个框架&quot;,而是&quot;我在标准刚发布时就做了企业适配&...
书海为家:#人脑vsAI# 尽管深度学习的最初灵感来源于人类的大脑,但二者的运作方式截然不同:深度学习所需要的数据量远比人脑所需要的多得多。可是一旦经过大数据训练,它在相同领域的表现将远远超过人类(尤其是在数字的量化学习,例如挑选某人最可能购买的产品,或从100万张脸中挑选最匹配的一张)——相对来说,人类在同一时间内只能把注意力放在少数几件事情上面,而深度学习算法却可以同时处理海量信息,并且发现在大量数据背后的模糊特征之间的关联,这些模糊特征不仅复杂而且微妙,人类往往无法理解,甚至可能不会注意到。 虽然深度学习拥有人类所缺乏的并行处理海量数据的“绝技”,但不具备人类在面对决策时独一无二的汲取过去的经验、使用抽象概念和常识的能力。 与人类相比,深度学习想要充分发挥作用,离不开海量的相关数据、单一领域的应用场景以及明确的目标函数,这三项缺一不可,如果缺少其中任何一项,深度学习将无用武之地。
AI求职实录
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务