類的內存對齊(類的sizeof)

1、空類的sizeof是1。空類是指沒有成員的類,類中的函數不佔空間,除非是虛函數。

如: class A

        {

             public:

                      A(){}

                     ~A(){}

                     void fun(){}

         };

sizeof(A)是1.

注: class A1

        {

             public:

                      A1(){}

                     ~A1(){}

                     void fun(){}

                       char a[0];

         };

sizeof(A1)也是1.(VC6.0下編譯)

2、若類中包含成員,則類對象的大小隻包括其中非靜態成員經過對齊所佔的空間,對齊方式和結構體相同。如:

class A
{
public:
  int b;
  float c;
  char d;
};

sizeof(A)是12.

class A
{
public:
  static int a;
  int b;
  float c;
  char d;
};
sizeof(A)是12.

class A
{
public:
  static int a;
  int b;
  float c;
  char d;
  int add(int x,int y)
  {
    return x+y;
  }
};
sizeof(A)也是12.

3、若類中包含虛函數,則無論有幾個虛函數,sizeof類都等於sizeof(數據成員)的和+sizeof(V表指針,爲4),如:

class Base
{
      public:
             Base(){cout<<"Base-ctor"<<endl;}
             ~Base(){cout<<"Base-dtor"<<endl;}
             int a;
             virtual void f(int) {cout<<"Base::f(int)"<<endl;}
             virtual void f(double){cout<<"Base::f(double)"<<endl;}
};

sizeof(Base)爲8.

4、對於子類,它的sizeof是它父類成員(無論成員是public或private),再加上它自己的成員,對齊後的sizeof,如:

class A2
{
      public:
             int a;
      private:
              char b;
};

class A3:public A2
{
      public:
             char b;
             short a;            
};

sizeof(A3)是8. 但如果A3如下:

class A3:public A2
{
      public:
             short a; 
             char b;          
};

sizeof(A3)是12.

5、對於子類和父類中都有虛函數的情況,子類的sizeof是它父類成員(無論成員是public或private),再加上它自己的成員,對齊後的sizeof,再加4(虛表指針)。如:

class Base
{
      public:
             Base(){cout<<"Base-ctor"<<endl;}
             ~Base(){cout<<"Base-dtor"<<endl;}
             int a;
             virtual void f(int) {cout<<"Base::f(int)"<<endl;}
             virtual void f(double){cout<<"Base::f(double)"<<endl;}
};

class Derived:public Base
{
  public:
         Derived(){cout<<"Derived-ctor"<<endl;}
         int b;
         virtual void g(int){cout<<"Derived::g(int)"<<endl;}
};

sizeof(Derived)是12.

6、對於虛繼承的子類,其sizeof的值是其父類成員,加上它自己的成員,以及它自己一個指向父類的指針(大小爲4),對齊後的sizeof。如:

#include   <iostream.h>  
   
  class   a  
  {  
  private:  
  int   x;  
  };  
   
   
   
  class   b:   virtual   public   a  
  {  
  private:  
  int   y;  
  };  
   
  class   c:   virtual   public   a  
  {  
  private:  
  int   z;  
  };  
   
  class   d:public   b,public   c  
  {  
  private:  
  int   m;  
  };  
  int   main(int   argc,   char*   argv[])  
  {  
  cout<<sizeof(a)<<endl;  
  cout<<sizeof(b)<<endl;  
  cout<<sizeof(c)<<endl;  
  cout<<sizeof(d)<<endl;  
  return   0;  
  }  
    在VC6.0下調試結果爲  
  4  
  12  
  12  
  24

sizeof(b)和sizeof(c)相同,都是4+4+4=12。

sizeof(d)是sizeof(b)(爲12)+sizeof(c)(爲12)-b和c相同的部分(a的成員,大小是4)+d自己的成員(大小爲4)=24

7、對於既有虛繼承又有虛函數的子類,其sizeof的值是其父類成員(計算虛表指針大小+4),加上它自己的成員(計算虛表指針大小+4),以及它自己一個指向父類的指針(大小爲4),對齊後的sizeof

class Base
{
public:
Base(){cout<<"Base-ctor"<<endl;}
~Base(){cout<<"Base-dtor"<<endl;}
virtual void f(int) {cout<<"Base::f(int)"<<endl;}
virtual void f(double){cout<<"Base::f(double)"<<endl;}
};

class Derived:virtual public Base
{
public:
Derived(){cout<<"Derived-ctor"<<endl;}
virtual void g(int){cout<<"Derived::g(int)"<<endl;}
};

sizeof(Base)=4

sizeof(Derived)=12 (父類虛表指針大小4+自己虛表指針大小4+子類指向父類的一個指針大小4=12)

對齊規則1
數據成員對齊規則:結構(struct)(或聯合(union))的數據成員,第一個數據成員放在offset爲0的地方,以後每個數據成員的對齊按照 #pragma pack指定的數值和這個數據成員自身長度中,比較小的那個進行。
a=數據成員自動長度
b=#pragma pack指定的數值n,n的取值範圍爲1,2,4,8,16
該數據成員對齊字節=min(a,b)

對齊規則2
結構(或聯合)的整體對齊規則:在數據成員完成各自對齊之後,結構(或聯合)本身也要進行對齊,對齊將按照#pragma pack指定的數值和結構(或聯合)最大數據成員長度中,比較小的那個進行。 
a=結構(或聯合)最大數據成員長度
b=#pragma pack指定的數值n,n的取值範圍爲1,2,4,8,16
該數據成員對齊字節=min(a.b)

對齊規則3
當#pragma pack的n值等於或超過所有數據成員長度的時候,這個n值的大小將不產生任何效果。
a=max(所有數據成員長度)
If(n>a){n無效}

對齊規則4
第一個數據成員偏移量爲0。其他各成員變量存放的起始地址相對於結構的起始地址的偏移量必須爲該變量的類型所佔用的字節數的倍數,倍數爲1。
a=0 ,a代表第一個數據成員偏移量
b=成員變量佔用字節數
n=偏移量
n%b=0

對齊規則5
各成員變量在存放的時候根據在結構中出現的順序依次申請空間,同時按照上面的對齊方式調整位置,空缺的字節自動填充。

對齊規則6
同時爲了確保結構的大小爲結構的字節邊界數(即該結構中佔用最大空間的類型所佔用的字節數)的倍數,所以在爲最後一個成員變量申請空間後,還會根據需要自動填充空缺的字節。
a=結構中佔用最大空間的類型所佔用的字節數
b=結構中除最後一個成員變量,其他成員所佔字節之和
結構體大小=大於b且是a的倍數的最小數。

對齊規則7
通過預編譯命令改變對齊係數爲1的時候,結構的大小等於結構體各個成員變量佔用字節之和。各成員變量依次存放,不產生填充字節。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章