C++ 對象和內(nèi)存

一個(gè)類創(chuàng)建的對象所占據(jù)的內(nèi)存空間是多少楔脯?
  1. 非靜態(tài)的數(shù)據(jù)成員的總和大小
  2. 邊界對齊(操作系統(tǒng)會根據(jù)計(jì)算機(jī)的位數(shù)臼予,將數(shù)據(jù)按照字長來對齊)
  3. 為了支持virtual機(jī)制鸣戴,而產(chǎn)生的額外負(fù)擔(dān)

Q1:一個(gè)空類的大小是多少?
class empty_class {
};

int main() {
    empty_class e;
      //輸出的結(jié)果是1
    cout << sizeof(e) << endl;
}

輸出的結(jié)果是1粘拾;
A1:也就是說一個(gè)空類的大小是1窄锅。

Q2:如何分析一個(gè)正常的類的大小缰雇?
class test_class {
private:
    int val;
};

int main(){
    test_class t;
     //輸出4
    cout << sizeof(t) << endl;
}

輸出的結(jié)果是4入偷,也就是說對象的大小是對象里面的數(shù)據(jù)成員的大小。

class test_class {
public:
    void fun2() {
        cout << "func2" << endl;
    }

    void fun3() {
        cout << "func3" << endl;
    }

    void fun1() {
        cout << "func1" << endl;
    }
};

int main(){
    test_class t;
    //輸出是1械哟,因?yàn)闆]有數(shù)據(jù)成員疏之,都是成員函數(shù),但是成員函數(shù)是不占據(jù)空間的
    cout << sizeof(t) << endl;
}

輸出是1暇咆,因?yàn)闆]有數(shù)據(jù)成員锋爪,都是成員函數(shù)丙曙,但是成員函數(shù)是不占據(jù)空間的


Q3:如何分析一個(gè)繼承類的對象的大小几缭?
class Animal {
private:
    int loc;
    int id;
public:
    void getid() {
        cout << id << endl;
    }

    void getloc() {
        cout << loc << endl;
    }

    void print() {
        cout << loc << endl;
        cout << id << endl;
    }
};

class Bear : public Animal {
private:
    char name[8];
    int age;

public:
    void getage() {
        cout << age << endl;
    }
};

int main(){
    Animal a;
    //a 里面有兩個(gè)int的數(shù)據(jù)成員河泳,大小是4*2
    cout << sizeof(a) << endl;

    Bear b;
    //基類對象的大小是8+ 派生類里面的大小 8*1+4 沃呢,也就是8+8+4=20
    cout << sizeof(b) << endl;
}
Q4:一個(gè)帶有虛函數(shù)的類的大小
class myclass {
public:
    int val;
    virtual void print() {
        cout << "123" << endl;
    }
};

int main() {
    myclass m;
    // 輸出的結(jié)果是8
    cout << sizeof(m) << endl;
}

注意這里輸出的結(jié)果是8年栓,為什么是8呢? 是因?yàn)槿绻粋€(gè)類包含了虛函數(shù)薄霜,那么創(chuàng)建對象的時(shí)候某抓,對象里面就會多一個(gè)虛函數(shù)的指針,所以是8惰瓜。但是如果去掉了virtual否副,輸出的結(jié)果就是4了。

這個(gè)其實(shí)告訴我們很重要的一點(diǎn):
對于一個(gè)類創(chuàng)建的對象崎坊,這個(gè)對象里面保存的不僅僅是數(shù)據(jù)成員备禀,還有一個(gè)就是虛函數(shù)指針。

所以奈揍,在構(gòu)造函數(shù)曲尸,拷貝構(gòu)造函數(shù),賦值構(gòu)造函數(shù)等等男翰,都需要考慮這個(gè)虛指針的情況另患。

Q5:體現(xiàn)內(nèi)存對齊的
class char_class {
public:
    char c[1];
    int val;
};

char_class ss;
//輸出的結(jié)果是 8
cout<<sizeof(ss)<<endl;

但是
class char_class {
public:
    char c[1];
};
char_class ss;
//輸出的結(jié)果是 1
cout<<sizeof(ss)<<endl;

這兩個(gè)不同的輸出,體現(xiàn)了內(nèi)存對齊的原則蛾绎。


對于有virtual的class的對象的分析

對于有virtual的class昆箕,每個(gè)對象里面還會有一個(gè)vptr(虛指針) 4個(gè)字節(jié)。

#include <iostream>
#include <typeinfo>
using namespace std;

typedef void(*Fun)(void);  //函數(shù)指針
// //關(guān)于為什么要這樣寫:https://www.cnblogs.com/windlaughing/archive/2013/04/10/3012012.html

class Parent 
{
public:
    int iparent;
    Parent() :iparent(10) {}
    virtual void f() { cout << " Parent::f()" << endl; }
    virtual void g() { cout << " Parent::g()" << endl; }
    virtual void h() { cout << " Parent::h()" << endl; }
};


class Child : public Parent 
{
public:
    int ichild;
    Child() :ichild(100) {}
    virtual void f() { cout << "Child::f()" << endl; }
    virtual void g_child() { cout << "Child::g_child()" << endl; }
    virtual void h_child() { cout << "Child::h_child()" << endl; }
};

class GrandChild : public Child 
{
public:
    int igrandchild;
    GrandChild() :igrandchild(1000) {}
    virtual void f() { cout << "GrandChild::f()" << endl; }
    virtual void g_child() { cout << "GrandChild::g_child()" << endl; }
    virtual void h_grandchild() { cout << "GrandChild::h_grandchild()" << endl; }
};

int main()
{
    Fun pFun = NULL;
    GrandChild gc;

    int** pVtab = (int**)&gc;
    cout << "[0] GrandChild::_vptr->" << endl;
    for (int i = 0; pVtab[0][i] != NULL; i++) 
    {
        pFun = (Fun)pVtab[0][i];
        cout << "    [" << i << "] ";
        pFun();
    }
    cout << "[1] Parent.iparent = " << (int)pVtab[1] << endl;
    cout << "[2] Child.ichild = " << (int)pVtab[2] << endl;
    cout << "[3] GrandChild.igrandchild = " << (int)pVtab[3] << endl;

    return 0;
}

這個(gè)對象的內(nèi)存布局租冠,如下:


這個(gè)就是這個(gè)對象的內(nèi)存布局.png
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末鹏倘,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子顽爹,更是在濱河造成了極大的恐慌第股,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,743評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件话原,死亡現(xiàn)場離奇詭異夕吻,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)繁仁,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,296評論 3 385
  • 文/潘曉璐 我一進(jìn)店門涉馅,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人黄虱,你說我怎么就攤上這事稚矿。” “怎么了?”我有些...
    開封第一講書人閱讀 157,285評論 0 348
  • 文/不壞的土叔 我叫張陵晤揣,是天一觀的道長桥爽。 經(jīng)常有香客問我,道長昧识,這世上最難降的妖魔是什么钠四? 我笑而不...
    開封第一講書人閱讀 56,485評論 1 283
  • 正文 為了忘掉前任,我火速辦了婚禮跪楞,結(jié)果婚禮上缀去,老公的妹妹穿的比我還像新娘。我一直安慰自己甸祭,他們只是感情好缕碎,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,581評論 6 386
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著池户,像睡著了一般咏雌。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上校焦,一...
    開封第一講書人閱讀 49,821評論 1 290
  • 那天赊抖,我揣著相機(jī)與錄音,去河邊找鬼斟湃。 笑死熏迹,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的凝赛。 我是一名探鬼主播注暗,決...
    沈念sama閱讀 38,960評論 3 408
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼墓猎!你這毒婦竟也來了捆昏?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,719評論 0 266
  • 序言:老撾萬榮一對情侶失蹤毙沾,失蹤者是張志新(化名)和其女友劉穎骗卜,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體左胞,經(jīng)...
    沈念sama閱讀 44,186評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡寇仓,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,516評論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了烤宙。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片遍烦。...
    茶點(diǎn)故事閱讀 38,650評論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖躺枕,靈堂內(nèi)的尸體忽然破棺而出服猪,到底是詐尸還是另有隱情供填,我是刑警寧澤,帶...
    沈念sama閱讀 34,329評論 4 330
  • 正文 年R本政府宣布罢猪,位于F島的核電站近她,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏膳帕。R本人自食惡果不足惜粘捎,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,936評論 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望备闲。 院中可真熱鬧晌端,春花似錦捅暴、人聲如沸恬砂。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,757評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽泻骤。三九已至,卻和暖如春梧奢,著一層夾襖步出監(jiān)牢的瞬間狱掂,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,991評論 1 266
  • 我被黑心中介騙來泰國打工亲轨, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留趋惨,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,370評論 2 360
  • 正文 我出身青樓惦蚊,卻偏偏與公主長得像器虾,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子蹦锋,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,527評論 2 349

推薦閱讀更多精彩內(nèi)容