国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > C++ > 正文

淺談C++ 類的實例中 內(nèi)存分配詳解

2020-05-23 13:54:21
字體:
供稿:網(wǎng)友

一個類,有成員變量:靜態(tài)與非靜態(tài)之分;而成員函數(shù)有三種:靜態(tài)的、非靜態(tài)的、虛的。

那么這些個東西在內(nèi)存中到底是如何分配的呢?

以一個例子來說明:

#include"iostream.h"class CObject{public:  static int a;  CObject();  ~CObject();  void Fun();private: int m_count; int m_index;};VoidCObject::Fun(){  cout<<"Fun/n"<<endl;}CObject::CObject(){  cout<<"Construct!/n";}CObject::~CObject(){  cout<<"Destruct!/n";}int CObject::a=1;void main(){cout<<"Sizeof(CObject):"<<sizeof(CObject)<<endl; cout<<"CObject::a="<<CObject::a<<endl;CObject myObject;cout<<"sizeof(myObject):"<<sizeof(myObject)<<endl;cout<<"sizeof(int)"<<sizeof(int)<<endl;}

 這是我的一段測試代碼, 運行結(jié)果是:

 

Sizeof(CObject):8CObject::a=1Construct!sizeof(myObject):8sizeof(int)4Destruct!

 我有疑問如下:

(1)C++中,應(yīng)該是對象才會被分配內(nèi)存空間吧??為什么CObject內(nèi)存大小是8,剛好和兩個成員變量的大小之和一致!難道還沒實例化的時候,類就已經(jīng)有了內(nèi)存空間了?

(2)當(dāng)對象生成了之后,算出的內(nèi)存大小怎么還是8,函數(shù)難道不占用內(nèi)存空間嗎?至少應(yīng)該放個函數(shù)指針在里面的吧?內(nèi)存是怎樣布局的?

(3)靜態(tài)成員應(yīng)該是屬于類的,怎么類的大小中沒有包含靜態(tài)成員的大小?

下面分別解答如下:

1)Sizeof(CObject)是在編譯時就計算了的,一個類定義了,它所占的內(nèi)存編譯器就已經(jīng)知道了,這時只是得到它占用的大小,并沒有分配內(nèi)存操作 。也可以這樣想:編譯器肯定知道大小了,這與分配內(nèi)存空間無關(guān),知道大小了,以后實例化了才能知道要分配多大。

2)類的普通成員、靜態(tài)成員函數(shù)是不占類內(nèi)存的,至于你說的函數(shù)指針在你的類中有虛函數(shù)的時候存在一個虛函數(shù)表指針,也就是說如果你的類里有虛函數(shù)則sizeof(CObject)的值會增加4個字節(jié)。

其實類的成員函數(shù)實際上與普通的全局函數(shù)一樣。

只不過編譯器在編譯的時候,會在成員函數(shù)上加一個參數(shù),傳入這個對象的指針。

成員函數(shù)地址是全局已知的,對象的內(nèi)存空間里根本無須保存成員函數(shù)地址。

對成員函數(shù)(非虛函數(shù))的調(diào)用在編譯時就確定了。

像 myObject.Fun() 這樣的調(diào)用會被編譯成形如 _CObject_Fun( &myObject ) 的樣子。

函數(shù)是不算到sizeof中的,因為函數(shù)是代碼,被各個對象共用,跟數(shù)據(jù)處理方式不同。對象中不必有函數(shù)指針,因為對象沒必要知道它的各個函數(shù)的地址(調(diào)用函數(shù)的是其他代碼而不是該對象)。

類的屬性是指類的數(shù)據(jù)成員,他們是實例化一個對象時就為數(shù)據(jù)成員分配內(nèi)存了,而且每個對象的數(shù)據(jù)成員是對立的,而成員函數(shù)是共有的~

靜態(tài)成員函數(shù)與一般成員函數(shù)的唯一區(qū)別就是沒有this指針,因此不能訪問非靜態(tài)數(shù)據(jù)成員。總之,程序中的所有函數(shù)都是位于代碼區(qū)的。

3)靜態(tài)成員并不屬于某個對象,sizeof取的是對象大小。

知道了上面的時候,就可以改一下來看看:

我也補充一些:

class CObject{public:static int a;CObject();~CObject();void Fun();private:double m_count; //這里改成了doubleint m_index;};這個類用sizeof()測出來的大小是 2*sizeof(double)=16 class CObject{public:static int a;CObject();~CObject();void Fun();private:char m_count; //這里改成了charint m_index;};大小是2*sizeof(int)=8class CObject{public:static int a;CObject();~CObject();void Fun();private:double m_count; //這里改成了doubleint m_index;char c;};sizeof(char)+sizeof(int) <sizeof(double) 所以大小是2*sizeof(double)

其實這里還有一個是內(nèi)存對齊的問題。

空類大小是1。

另外要注意的一些問題:

先看一個空的類占多少空間?

class Base { public:   Base();   ~Base(); };  class Base { public:Base(); ~Base(); };

注意到我這里顯示聲明了構(gòu)造跟析構(gòu),但是sizeof(Base)的結(jié)果是1.

因為一個空類也要實例化,所謂類的實例化就是在內(nèi)存中分配一塊地址,每個實例在內(nèi)存中都有獨一無二的地址。同樣空類也會被實例化,所以編譯器會給空類隱含 的添加一個字節(jié),這樣空類實例化之后就有了獨一無二的地址了。所以空類的sizeof為1。

而析構(gòu)函數(shù),跟構(gòu)造函數(shù)這些成員函數(shù),是跟sizeof無關(guān)的,也不難理解因為我們的sizeof是針對實例,而普通成員函數(shù),是針對類體的,一個類的成 員函數(shù),多個實例也共用相同的函數(shù)指針,所以自然不能歸為實例的大小,這在我的另一篇博文有提到。

接著看下面一段代碼

class Base { public:   Base();           virtual ~Base();     //每個實例都有虛函數(shù)表   void set_num(int num)  // 普通成員函數(shù),為各實例公有,不歸入sizeof統(tǒng)計   {     a=num;   } private:   int a;         //占4字節(jié)   char *p;         //4字節(jié)指針 };   class Derive:public Base { public:   Derive():Base(){};     ~Derive(){}; private:   static int st;     //非實例獨占   int d;           //占4字節(jié)   char *p;          //4字節(jié)指針 };  int main() {   cout<<sizeof(Base)<<endl;   cout<<sizeof(Derive)<<endl;   return 0; }  class Base { public:Base();virtual ~Base(); //每個實例都有虛函數(shù)表void set_num(int num) { a=num; } //普通成員函數(shù),為各實例公有,不歸入sizeof統(tǒng)計private: int a; //占4字節(jié)char *p; //4字節(jié)指針};class Derive:public Base {public:Derive():Base(){};~Derive(){};private:static int st; //非實例獨占int d; //占4字節(jié)char *p; //4字節(jié)指針};int main() {cout<<sizeof(Base)<<endl;cout<<sizeof(Derive)<<endl; return 0;}

結(jié)果自然是

12

20

Base類里的int  a;char *p;占8個字節(jié)。

而虛析構(gòu)函數(shù)virtual ~Base();的指針占4子字節(jié)。

其他成員函數(shù)不歸入sizeof統(tǒng)計。

Derive類首先要具有Base類的部分,也就是占12字節(jié)。

int  d;char *p;占8字節(jié)

static int st;不歸入sizeof統(tǒng)計

所以一共是20字節(jié)。

在考慮在Derive里加一個成員char c;

class Derive:public Base { public:   Derive():Base(){};   ~Derive(){}; private:   static int st;   int d;   char *p;   char c; };  class Derive:public Base { public:  Derive():Base(){};  ~Derive(){}; private: static int st;  int d; char *p;  char c; };

這個時候,結(jié)果就變成了

12

24

一個char c;增加了4字節(jié),說明類的大小也遵守類似class字節(jié)對齊,補齊規(guī)則。

至此,我們可以歸納以下幾個原則:

1.類的大小為類的非靜態(tài)成員數(shù)據(jù)的類型大小之和,也就是說靜態(tài)成員數(shù)據(jù)不作考慮。

2.普通成員函數(shù)與sizeof無關(guān)。

以上就是小編為大家?guī)淼臏\談C++ 類的實例中 內(nèi)存分配詳解全部內(nèi)容了,希望大家多多支持VEVB武林網(wǎng)~


發(fā)表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發(fā)表
主站蜘蛛池模板: 清镇市| 上栗县| 大名县| 卓尼县| 威海市| 巴彦县| 长宁区| 潢川县| 资中县| 札达县| 丘北县| 海南省| 旌德县| 湘乡市| 沧源| 雅江县| 伊川县| 锡林浩特市| 镇远县| 麻城市| 渝北区| 张家港市| 沈阳市| 桂平市| 家居| 顺昌县| 乌苏市| 屏东市| 靖江市| 楚雄市| 甘德县| 清原| 会宁县| 汝州市| 巴林右旗| 永康市| 泌阳县| 凤冈县| 临高县| 鲁山县| 镶黄旗|