亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

淺談C++ 類的實例中 內存分配詳解

 更新時間:2016年12月27日 11:29:41   投稿:jingxian  
下面小編就為大家?guī)硪黄獪\談C++ 類的實例中 內存分配詳解。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧

一個類,有成員變量:靜態(tài)與非靜態(tài)之分;而成員函數(shù)有三種:靜態(tài)的、非靜態(tà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;
}

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

 

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

 我有疑問如下:

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

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

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

下面分別解答如下:

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

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

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

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

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

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

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

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

類的屬性是指類的數(shù)據(jù)成員,他們是實例化一個對象時就為數(shù)據(jù)成員分配內存了,而且每個對象的數(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; //這里改成了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(); 
}; 
 class Base {
 public:
Base();
 ~Base();
 };

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

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

而析構函數(shù),跟構造函數(shù)這些成員函數(shù),是跟sizeof無關的,也不難理解因為我們的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;
}

結果自然是

12

20

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

而虛析構函數(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;
 
};

這個時候,結果就變成了

12

24

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

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

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

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

以上就是小編為大家?guī)淼臏\談C++ 類的實例中 內存分配詳解全部內容了,希望大家多多支持腳本之家~

相關文章

  • 關于STL中l(wèi)ist容器的一些總結

    關于STL中l(wèi)ist容器的一些總結

    list就是數(shù)據(jù)結構中的雙向鏈表(根據(jù)sgi stl源代碼),因此它的內存空間是不連續(xù)的,通過指針來進行數(shù)據(jù)的訪問,這個特點使得它的隨即存取變的非常沒有效率,因此它沒有提供[]操作符的重載
    2013-09-09
  • typedef和#define用法區(qū)別總結

    typedef和#define用法區(qū)別總結

    在C還是C++代碼中,typedef都使用的很多,在C代碼中尤其多,typedef與#define有些相似,其實是不同的,特別是在一些復雜的用法上,下面這篇文章主要給大家介紹了關于typedef和#define用法區(qū)別總結的相關資料,需要的朋友可以參考下
    2023-06-06
  • Qt6.0?qproperty-*不生效原因解決分析

    Qt6.0?qproperty-*不生效原因解決分析

    這篇文章主要為大家介紹了Qt6.0?qproperty-*不生效原因解決分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-08-08
  • C++單例類模板詳解

    C++單例類模板詳解

    這篇文章主要介紹了C++單例類模板,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2019-03-03
  • 數(shù)據(jù)結構 雙機調度問題的實例詳解

    數(shù)據(jù)結構 雙機調度問題的實例詳解

    這篇文章主要介紹了數(shù)據(jù)結構 雙機調度問題的實例詳解的相關資料,雙機調度問題,又稱獨立任務最優(yōu)調度:用兩臺處理機A和B處理n個作業(yè)的實例,需要的朋友可以參考下
    2017-08-08
  • 從頭學習C語言之字符串處理函數(shù)

    從頭學習C語言之字符串處理函數(shù)

    這篇文章主要為大家詳細介紹了C語言之字符串處理函數(shù),文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下,?希望能夠給你帶來幫助
    2022-01-01
  • CMakeList中自動編譯protobuf文件過程

    CMakeList中自動編譯protobuf文件過程

    這篇文章主要介紹了CMakeList中自動編譯protobuf文件過程,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-12-12
  • C++中智能指針unique_ptr的實現(xiàn)詳解

    C++中智能指針unique_ptr的實現(xiàn)詳解

    智能指針本質上并不神秘,其實就是?RAII?資源管理功能的自然展現(xiàn)而已,這篇文章主要為大家詳細介紹了如何實現(xiàn)?C++中智能指針的?unique_ptr,需要的可以了解下
    2024-01-01
  • 一文帶你搞懂C++中的流量控制

    一文帶你搞懂C++中的流量控制

    限流可以認為服務降級的一種,限流就是限制系統(tǒng)的輸入和輸出流量已達到保護系統(tǒng)的目的,這篇文章小編就來帶大家深入了解一下如何利用C++實現(xiàn)流量控制吧
    2023-10-10
  • C++模擬實現(xiàn)list功能

    C++模擬實現(xiàn)list功能

    list的底層是一個循環(huán)雙向鏈表結構,雙向鏈表中每個元素存儲在互不相關的獨立節(jié)點中,在節(jié)點中通過指針指向其前一個元素和后一個元素,接下來通過本文給大家分享C++模擬實現(xiàn)list的示例代碼,需要的朋友可以參考下
    2021-08-08

最新評論