c++對象模型--虛表

c++對象模型

首先看這樣一個(gè)類

class Test{

};
TestSize.png
  • 在c++中,對于一個(gè)空類,c++不允許什么都不存在肺孤,會有一個(gè)字節(jié)的占位空間替蛉。

如果類中有成員變量

class Test{
    char a[2];
};
TestSizeChar.png
  • 現(xiàn)在看似很正常

如果類中有函數(shù)

class Test{
    public:
        void fun()
        {
            
        }
};
TestSize.png

如果類中有虛函數(shù)

class Test{
    public:
        virtual void fun()
        {

        }
};
TestSizeVirtual.png
  • 此時(shí)谷遂,c++會在類的中產(chǎn)生一個(gè)指針,這是一個(gè)函數(shù)指針,指向虛函數(shù)--虛表,因?yàn)樘摵瘮?shù)要實(shí)現(xiàn)一種動態(tài)綁定挤渔,而其他函數(shù)不需要,所以风题,有虛函數(shù)存在的時(shí)候判导,類的對象中會有一個(gè)指針來指向所有虛函數(shù)嫉父。

如果類中有虛函數(shù)和成員變量

class Test{
    public:
        char a[2];
        virtual void fun(){}
};
TestSizeVirtualChar.png
  • 按理說sizeof(Test)應(yīng)該是2+8=10,但是卻是16眼刃,這是因?yàn)槿葡剑藭r(shí)會發(fā)生內(nèi)存對齊,為了效率擂红,會讓內(nèi)存的大小按照最大的字段的整數(shù)倍進(jìn)行對齊仪际。

驗(yàn)證虛表

class TestModel{
    public:
        virtual void test(){
            cout<<"this="<<this<<endl;
            cout<<"test"<<endl;
            my();
        }
        virtual void test1(){
            cout<<"this="<<this<<endl;
            cout<<"test1"<<endl;
            myTest();
            
        }
        TestModel(){
            cout<<"構(gòu)造"<<endl;
        }
        virtual ~TestModel(){
            cout<<"析構(gòu)"<<endl;
        }
        int i;
    private:
        void myTest(){
            cout<<"myTest"<<endl;
        } 
        void my(){
            cout<<"my"<<i<<endl;
        }
};

對于上面這樣一個(gè)類,他的內(nèi)存分布情況如下

+---------+
| other   |---> 其他字段
|---------|
| vptr    |---> 虛表 ------->       
+---------+                 +-----------------------------+
                           0|  test(TestModel* this)      |
                            |-----------------------------|
                           1|  test1(TestModel* this)     |
                            |-----------------------------|
                           2|  ~TestModel(TestModel* this)|
                            +-----------------------------+

測試代碼

#include <iostream>
#include <stdio.h>

using namespace std;

class TestModel{
    public:
        virtual void test(){
            cout<<"this="<<this<<endl;
            cout<<"test"<<endl;
            my();
        }
        virtual void test1(){
            cout<<"this="<<this<<endl;
            cout<<"test1"<<endl;
            myTest();
            
        }
        TestModel(){
            cout<<"構(gòu)造"<<endl;
        }
        virtual ~TestModel(){
            cout<<"析構(gòu)"<<endl;
        }
        int i;
    private:
        void myTest(){
            cout<<"myTest"<<endl;
        } 
        void my(){
            cout<<"my"<<i<<endl;
        }
};
typedef void (*Fun)(void* self);
void fun(){
    cout<<"fun"<<endl;
}

int main(int argc, char const *argv[])
{
    
    TestModel a;
    TestModel b;
    a.i=0;
    b.i=1;
    int64_t* pa = (int64_t*)&a;
    int64_t* pb = (int64_t*)&b;
    void* p1;
    printf("指針大小%d\n",sizeof(void*));
    printf("a的地址%p\n",pa);
    printf("a的虛表地址%p\n",*pa);
    printf("b的地址%p\n",pb);
    printf("b的虛表地址%p\n",*pa);

    Fun funp;
    funp = (Fun)(*((int64_t*)(*(pa))));
    funp(pa);
    funp = (Fun)(*((int64_t*)(*(pa))+1));
    funp(NULL);
    funp = (Fun)(*((int64_t*)(*(pa))+2));
    funp(pa);
    
    return 0;
}

TestSizeMain.png
+---------+
| vptr    |---> 虛表 ------->                              <---*pa==*pb
|---------|                  +-----------------------------+
| other   |                 0|  test(TestModel* this)      | *((int64_t*)(*(pa)))
+---------+                  |-----------------------------|
                            1|  test1(TestModel* this)     | *((int64_t*)(*(pa))+1)
                             |-----------------------------|
                            2|  ~TestModel(TestModel* this)| *((int64_t*)(*(pa))+2)
                             +-----------------------------+
  • 測試代碼中篮条,用一個(gè)函數(shù)指針指向虛表的各個(gè)函數(shù),就能調(diào)用到對應(yīng)的函數(shù)吩抓,說明正確的找到了函數(shù)
  • 并且通過傳入的函數(shù)的指針值不同涉茧,調(diào)用的函數(shù)中打印出來的i不同,可以說明this指針是在函數(shù)調(diào)用的時(shí)候通過參數(shù)傳入的
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末疹娶,一起剝皮案震驚了整個(gè)濱河市伴栓,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌雨饺,老刑警劉巖钳垮,帶你破解...
    沈念sama閱讀 212,454評論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異额港,居然都是意外死亡饺窿,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,553評論 3 385
  • 文/潘曉璐 我一進(jìn)店門移斩,熙熙樓的掌柜王于貴愁眉苦臉地迎上來肚医,“玉大人,你說我怎么就攤上這事向瓷〕μ祝” “怎么了?”我有些...
    開封第一講書人閱讀 157,921評論 0 348
  • 文/不壞的土叔 我叫張陵猖任,是天一觀的道長你稚。 經(jīng)常有香客問我,道長朱躺,這世上最難降的妖魔是什么刁赖? 我笑而不...
    開封第一講書人閱讀 56,648評論 1 284
  • 正文 為了忘掉前任,我火速辦了婚禮长搀,結(jié)果婚禮上乾闰,老公的妹妹穿的比我還像新娘。我一直安慰自己盈滴,他們只是感情好涯肩,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,770評論 6 386
  • 文/花漫 我一把揭開白布轿钠。 她就那樣靜靜地躺著,像睡著了一般病苗。 火紅的嫁衣襯著肌膚如雪疗垛。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,950評論 1 291
  • 那天硫朦,我揣著相機(jī)與錄音贷腕,去河邊找鬼。 笑死咬展,一個(gè)胖子當(dāng)著我的面吹牛泽裳,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播破婆,決...
    沈念sama閱讀 39,090評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼涮总,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了祷舀?” 一聲冷哼從身側(cè)響起瀑梗,我...
    開封第一講書人閱讀 37,817評論 0 268
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎裳扯,沒想到半個(gè)月后抛丽,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,275評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡饰豺,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,592評論 2 327
  • 正文 我和宋清朗相戀三年亿鲜,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片冤吨。...
    茶點(diǎn)故事閱讀 38,724評論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡狡门,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出锅很,到底是詐尸還是另有隱情其馏,我是刑警寧澤,帶...
    沈念sama閱讀 34,409評論 4 333
  • 正文 年R本政府宣布爆安,位于F島的核電站叛复,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏扔仓。R本人自食惡果不足惜褐奥,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,052評論 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望翘簇。 院中可真熱鬧撬码,春花似錦、人聲如沸版保。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,815評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至叫胁,卻和暖如春凰慈,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背驼鹅。 一陣腳步聲響...
    開封第一講書人閱讀 32,043評論 1 266
  • 我被黑心中介騙來泰國打工微谓, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人输钩。 一個(gè)月前我還...
    沈念sama閱讀 46,503評論 2 361
  • 正文 我出身青樓豺型,卻偏偏與公主長得像,于是被迫代替她去往敵國和親买乃。 傳聞我的和親對象是個(gè)殘疾皇子姻氨,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,627評論 2 350

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