类中内容在内存中到底是如何分配的呢?
??????一個類,有成員變量:靜態與非靜態之分;而成員函數有三種:靜態的、非靜態的、虛的。
??????那么這些個東西在內存中到底是如何分配的呢?
??????以一個例子來說明:
??????[html]view plaincopyprint?
運行結果是:?
Sizeof(CObject):8?
CObject::a=1?
Construct!?
sizeof(myObject):8?
sizeof(int)4?
Destruct!?
我有疑問如下:?
(1) C++中,應該是對象才會被分配內存空間吧??為什么CObject內存大小是8,剛好和兩個 成員變量的大小之和一致!難道還沒實例化的時候,類就 已經有了內存空間了??
(2)當對象生成了之后,算出的內存大小怎么還是8,函數難道不占用內存空間嗎?至少應該放個 函數指針在里面的吧?內存是怎樣布局的?
(3) 靜態成員應該是屬于類的,怎么類的大小中沒有包含靜態成員的大小?
下面分別解答如下:
1)Sizeof(CObject)是在編譯時就計算了的,一個類定義了,它所占的內存編譯器就已經知道了,這時只是得到它占用的大小,并沒有分配內存操作 。也可以這樣想:編譯器肯定知道大小了,這與分配內存空間無關,知道大小了,以后實例化了才能知道要分配多大。
2)類的普通成員、靜態成員函數是不占類內存的,至于你說的函數指針在你的類中有 虛函數的時候存在一個 虛函數表指針,也就是說如果你的類里有虛函數則 sizeof(CObject)的值會增加4個字節。
其實 類的成員函數 實際上與 普通的 全局函數一樣。?
只不過編譯器在編譯的時候,會在成員函數上加一個參數,傳入這個對象的指針。
成員函數地址是全局已知的,對象的內存空間里根本無須保存成員函數地址。?
對成員函數(非虛函數)的調用在編譯時就確定了。?
像 myObject.Fun() 這樣的調用會被編譯成形如 _CObject_Fun( &myObject ) 的樣子。
函數是不算到sizeof中的,因為函數是代碼,被各個對象共用,跟數據處理方式不同。對象中不必有函數指針,因為對象沒必要知道它的各個函數的地址(調 用函數的是其他代碼而不是該對象)。?
類的屬性是指類的數據成員,他們是實例化一個對象時就為數據成員分配內存了,而且每個對象的數據成員是對立的,而成員函數是共有的~?
靜態成員函數與一般成員函數的唯一區別就是沒有this指針,因此不能訪問非靜態數據成員。總之,程序中的所有函數都是位于代碼區的。
3)靜態成員并不屬于某個對象,sizeof取的是對象大小。
知道了上面的時候,就可以改一下來看看:
我也補充一些:?
class CObject?
{?
public:?
static int a;?
CObject();?
~CObject();?
void Fun();?
private:?
double m_count;? //這里改成了double?
int? m_index;?
};?
這個類用sizeof()測出來的大小是 2*sizeof(double)=16?
class CObject?
{?
public:?
static int a;?
CObject();?
~CObject();?
void Fun();?
private:?
char m_count;? //這里改成了char?
int? m_index;?
};?
大小是2*sizeof(int)=8?
class CObject?
{?
public:?
static int a;?
CObject();?
~CObject();?
void Fun();?
private:?
double m_count;? //這里改成了double?
int? m_index;?
char? c;?
};?
sizeof(char)+sizeof(int) <sizeof(double) 所以大小是2*sizeof(double)?
其實這里還有一個是內存對齊的問題。
空類大小是1。?
另外要注意的一些問題:
??????先看一個空的類占多少空間?
??????class?Base
??????{
??????public:
??????Base();
??????~Base();
??????};?
??????注意到我這里顯示聲明了構造跟析構,但是sizeof(Base)的結果是1.
??????因為一個空類也要實例化,所謂類的實例化就是在內存中分配一塊地址,每個實例在內存中都有獨一無二的地址。同樣空類也會被實例化,所以編譯器會給空類隱含 的添加一個字節,這樣空類實例化之后就有了獨一無二的地址了。所以空類的sizeof為1。
??????而析構函數,跟構造函數這些成員函數,是跟sizeof無關的,也不難理解因為我們的sizeof是針對實例,而普通成員函數,是針對類體的,一個類的成 員函數,多個實例也共用相同的函數指針,所以自然不能歸為實例的大小,這在我的另一篇博文有提到。
??????接著看下面一段代碼
??????[html]view plaincopyprint?
??????結果自然是
??????12
??????20
??????Base類里的int? a;char *p;占8個字節。
??????而虛析構函數virtual ~Base();的指針占4子字節。
??????其他成員函數不歸入sizeof統計。
??????Derive類首先要具有Base類的部分,也就是占12字節。
??????int? d;char *p;占8字節
??????static int st;不歸入sizeof統計
??????所以一共是20字節。
??????在考慮在Derive里加一個成員char c;
??????這個時候,結果就變成了
??????12
??????24
??????一個char c;增加了4字節,說明類的大小也遵守類似class字節對齊,的補齊規則。
??????具體的可以看我那篇《5分鐘搞定字節對齊》
??????至此,我們可以歸納以下幾個原則:
??????1.類的大小為類的非靜態成員數據的類型大小之和,也 就是說靜態成員數據不作考慮。
??????2.普通成員函數與sizeof無關。
??????3.虛函數由于要維護在虛函數表,所以要占據一個指針大小,也就是4字節。
??????4.類的總大小也遵守類似class字節對齊的,調整規則。
??????轉載自:http://www.blue1000.com/bkhtml/c151/2010-11/69613.htm
總結
以上是生活随笔為你收集整理的类中内容在内存中到底是如何分配的呢?的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 不能干一辈子开发???
- 下一篇: 什么变量在堆内存里存放,什么变量在栈内存