青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

yehao's Blog

C++ 對(duì)象的內(nèi)存布局(上)

轉(zhuǎn)自http://blog.csdn.net/haoel/archive/2008/10/15/3081328.aspx

前言
 

07年12月,我寫了一篇《C++虛函數(shù)表解析》的文章,引起了大家的興趣。有很多朋友對(duì)我的文章留了言,有鼓勵(lì)我的,有批評(píng)我的,還有很多問(wèn)問(wèn)題的。我在這里一并對(duì)大家的留言表示感謝。這也是我為什么再寫一篇續(xù)言的原因。因?yàn)椋谏弦黄恼轮校矣昧说氖纠际欠浅:?jiǎn)單的,主要是為了說(shuō)明一些機(jī)理上的問(wèn)題,也是為了圖一些表達(dá)上方便和簡(jiǎn)單。不想,這篇文章成為了打開C++對(duì)象模型內(nèi)存布局的一個(gè)引子,引發(fā)了大家對(duì)C++對(duì)象的更深層次的討論。當(dāng)然,我之前的文章還有很多方面沒(méi)有涉及,從我個(gè)人感覺(jué)下來(lái),在談?wù)撎摵瘮?shù)表里,至少有以下這些內(nèi)容沒(méi)有涉及:

 

1)有成員變量的情況。

2)有重復(fù)繼承的情況。

3)有虛擬繼承的情況。

4)有鉆石型虛擬繼承的情況。

 

這些都是我本篇文章需要向大家說(shuō)明的東西。所以,這篇文章將會(huì)是《C++虛函數(shù)表解析》的一個(gè)續(xù)篇,也是一篇高級(jí)進(jìn)階的文章。我希望大家在讀這篇文章之前對(duì)C++有一定的基礎(chǔ)和了解,并能先讀我的上一篇文章。因?yàn)檫@篇文章的深度可能會(huì)比較深,而且會(huì)比較雜亂,我希望你在讀本篇文章時(shí)不會(huì)有大腦思維紊亂導(dǎo)致大腦死機(jī)的情況。;-)

 

對(duì)象的影響因素
 

簡(jiǎn)而言之,我們一個(gè)類可能會(huì)有如下的影響因素:

 

1)成員變量

2)虛函數(shù)(產(chǎn)生虛函數(shù)表)

3)單一繼承(只繼承于一個(gè)類)

4)多重繼承(繼承多個(gè)類)

5)重復(fù)繼承(繼承的多個(gè)父類中其父類有相同的超類)

6)虛擬繼承(使用virtual方式繼承,為了保證繼承后父類的內(nèi)存布局只會(huì)存在一份)

上述的東西通常是C++這門語(yǔ)言在語(yǔ)義方面對(duì)對(duì)象內(nèi)部的影響因素,當(dāng)然,還會(huì)有編譯器的影響(比如優(yōu)化),還有字節(jié)對(duì)齊的影響。在這里我們都不討論,我們只討論C++語(yǔ)言上的影響。

 

本篇文章著重討論下述幾個(gè)情況下的C++對(duì)象的內(nèi)存布局情況。

 

1)單一的一般繼承(帶成員變量、虛函數(shù)、虛函數(shù)覆蓋)

2)單一的虛擬繼承(帶成員變量、虛函數(shù)、虛函數(shù)覆蓋)

3)多重繼承(帶成員變量、虛函數(shù)、虛函數(shù)覆蓋)

4)重復(fù)多重繼承(帶成員變量、虛函數(shù)、虛函數(shù)覆蓋)

5)鉆石型的虛擬多重繼承(帶成員變量、虛函數(shù)、虛函數(shù)覆蓋)

 

我們的目標(biāo)就是,讓事情越來(lái)越復(fù)雜。

 

知識(shí)復(fù)習(xí)
 

我們簡(jiǎn)單地復(fù)習(xí)一下,我們可以通過(guò)對(duì)象的地址來(lái)取得虛函數(shù)表的地址,如:

 

          typedef void(*Fun)(void);
 

            Base b;
 

            Fun pFun = NULL;
 

            cout << "虛函數(shù)表地址:" << (int*)(&b) << endl;
            cout << "虛函數(shù)表 — 第一個(gè)函數(shù)地址:" << (int*)*(int*)(&b) << endl;
 

            // Invoke the first virtual function 
            pFun = (Fun)*((int*)*(int*)(&b));
            pFun();
 

我們同樣可以用這種方式來(lái)取得整個(gè)對(duì)象實(shí)例的內(nèi)存布局。因?yàn)檫@些東西在內(nèi)存中都是連續(xù)分布的,我們只需要使用適當(dāng)?shù)牡刂菲屏浚覀兙涂梢垣@得整個(gè)內(nèi)存對(duì)象的布局。

 

本篇文章中的例程或內(nèi)存布局主要使用如下編譯器和系統(tǒng):

1)Windows XP 和 VC++ 2003

2)Cygwin 和 G++ 3.4.4

 

單一的一般繼承
 

下面,我們假設(shè)有如下所示的一個(gè)繼承關(guān)系:


請(qǐng)注意,在這個(gè)繼承關(guān)系中,父類,子類,孫子類都有自己的一個(gè)成員變量。而了類覆蓋了父類的f()方法,孫子類覆蓋了子類的g_child()及其超類的f()。

 

我們的源程序如下所示:

 

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; }
};
我們使用以下程序作為測(cè)試程序:(下面程序中,我使用了一個(gè)int** pVtab 來(lái)作為遍歷對(duì)象內(nèi)存布局的指針,這樣,我就可以方便地像使用數(shù)組一樣來(lái)遍歷所有的成員包括其虛函數(shù)表了,在后面的程序中,我也是用這樣的方法的,請(qǐng)不必感到奇怪,)

 

    typedef void(*Fun)(void);

 

    GrandChild gc;
   
 

    int** pVtab = (int**)&gc;
 

    cout << "[0] GrandChild::_vptr->" << endl;
    for (int i=0; (Fun)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;
 

其運(yùn)行結(jié)果如下所示:(在VC++ 2003和G++ 3.4.4下)

 

 

使用圖片表示如下:


可見以下幾個(gè)方面:

1)虛函數(shù)表在最前面的位置。

2)成員變量根據(jù)其繼承和聲明順序依次放在后面。

3)在單一的繼承中,被overwrite的虛函數(shù)在虛函數(shù)表中得到了更新。

多重繼承

 

下面,再讓我們來(lái)看看多重繼承中的情況,假設(shè)有下面這樣一個(gè)類的繼承關(guān)系。注意:子類只overwrite了父類的f()函數(shù),而還有一個(gè)是自己的函數(shù)(我們這樣做的目的是為了用g1()作為一個(gè)標(biāo)記來(lái)標(biāo)明子類的虛函數(shù)表)。而且每個(gè)類中都有一個(gè)自己的成員變量:

 


我們的類繼承的源代碼如下所示:父類的成員初始為10,20,30,子類的為100

 

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

};
 

class Base2 {
public:
    int ibase2;
    Base2():ibase2(20) {}
    virtual void f() { cout << "Base2::f()" << endl; }
    virtual void g() { cout << "Base2::g()" << endl; }
    virtual void h() { cout << "Base2::h()" << endl; }
};
 

class Base3 {
public:
    int ibase3;
    Base3():ibase3(30) {}
    virtual void f() { cout << "Base3::f()" << endl; }
    virtual void g() { cout << "Base3::g()" << endl; }
    virtual void h() { cout << "Base3::h()" << endl; }
};
 

 

class Derive : public Base1, public Base2, public Base3 {
public:
    int iderive;
    Derive():iderive(100) {}
    virtual void f() { cout << "Derive::f()" << endl; }
    virtual void g1() { cout << "Derive::g1()" << endl; }
};

 

我們通過(guò)下面的程序來(lái)查看子類實(shí)例的內(nèi)存布局:下面程序中,注意我使用了一個(gè)s變量,其中用到了sizof(Base)來(lái)找下一個(gè)類的偏移量。(因?yàn)槲衣暶鞯氖莍nt成員,所以是4個(gè)字節(jié),所以沒(méi)有對(duì)齊問(wèn)題。關(guān)于內(nèi)存的對(duì)齊問(wèn)題,大家可以自行試驗(yàn),我在這里就不多說(shuō)了)

 

             typedef void(*Fun)(void);


               Derive d;
 

                int** pVtab = (int**)&d;
 

                cout << "[0] Base1::_vptr->" << endl;
                pFun = (Fun)pVtab[0][0];
                cout << "     [0] ";
                pFun();
 

                pFun = (Fun)pVtab[0][1];
                cout << "     [1] ";pFun();
 

                pFun = (Fun)pVtab[0][2];
                cout << "     [2] ";pFun();
 

                pFun = (Fun)pVtab[0][3];
                cout << "     [3] "; pFun();
 

                pFun = (Fun)pVtab[0][4];
                cout << "     [4] "; cout<<pFun<<endl;
 

                cout << "[1] Base1.ibase1 = " << (int)pVtab[1] << endl;
 

 

                int s = sizeof(Base1)/4;
 

                cout << "[" << s << "] Base2::_vptr->"<<endl;
                pFun = (Fun)pVtab[s][0];
                cout << "     [0] "; pFun();
 

                Fun = (Fun)pVtab[s][1];
                cout << "     [1] "; pFun();
 

                pFun = (Fun)pVtab[s][2];
                cout << "     [2] "; pFun();
 

                pFun = (Fun)pVtab[s][3];
                out << "     [3] ";
                cout<<pFun<<endl;
 

                cout << "["<< s+1 <<"] Base2.ibase2 = " << (int)pVtab[s+1] << endl;
 

                s = s + sizeof(Base2)/4;


                cout << "[" << s << "] Base3::_vptr->"<<endl;
                pFun = (Fun)pVtab[s][0];
                cout << "     [0] "; pFun();
 

                pFun = (Fun)pVtab[s][1];
                cout << "     [1] "; pFun();
 

                pFun = (Fun)pVtab[s][2];
                cout << "     [2] "; pFun();
 

                pFun = (Fun)pVtab[s][3];
                 cout << "     [3] ";
                cout<<pFun<<endl;
 

                s++;
                cout << "["<< s <<"] Base3.ibase3 = " << (int)pVtab[s] << endl;
                s++;
                cout << "["<< s <<"] Derive.iderive = " << (int)pVtab[s] << endl;

 

其運(yùn)行結(jié)果如下所示:(在VC++ 2003和G++ 3.4.4下)

使用圖片表示是下面這個(gè)樣子:

 

我們可以看到:

1) 每個(gè)父類都有自己的虛表。

2) 子類的成員函數(shù)被放到了第一個(gè)父類的表中。

3) 內(nèi)存布局中,其父類布局依次按聲明順序排列。

4) 每個(gè)父類的虛表中的f()函數(shù)都被overwrite成了子類的f()。這樣做就是為了解決不同的父類類型的指針指向同一個(gè)子類實(shí)例,而能夠調(diào)用到實(shí)際的函數(shù)。

posted on 2011-04-24 19:15 厚積薄發(fā) 閱讀(257) 評(píng)論(0)  編輯 收藏 引用 所屬分類: C/C++

導(dǎo)航

<2025年11月>
2627282930311
2345678
9101112131415
16171819202122
23242526272829
30123456

統(tǒng)計(jì)

常用鏈接

留言簿

隨筆分類

文章分類

文章檔案

搜索

最新評(píng)論

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            亚洲天堂网在线观看| 亚洲资源av| 国内成人在线| 亚洲天堂男人| 亚洲精品综合精品自拍| 久久久91精品| 久久久国产精品一区二区中文 | 亚洲黄页视频免费观看| 亚洲欧美不卡| 午夜精品久久久久99热蜜桃导演| 女人天堂亚洲aⅴ在线观看| 久久免费黄色| 国产视频在线一区二区| 亚洲中午字幕| 午夜精品久久久久久久99水蜜桃| 欧美日韩一区二区三区四区在线观看| 欧美国产日韩a欧美在线观看| 国产午夜精品在线| 性欧美超级视频| 久久av一区二区三区漫画| 国产精品丝袜白浆摸在线| 99精品99| 亚洲综合电影| 国产精品亚洲成人| 午夜在线精品| 久久久综合网站| 狠狠色丁香婷婷综合影院| 久久国产天堂福利天堂| 久久久久国色av免费看影院| 国产亚洲美州欧州综合国| 性18欧美另类| 久久综合久久久| 亚洲第一页自拍| 欧美激情精品久久久久久久变态| 欧美高清在线观看| 亚洲美女毛片| 欧美视频在线一区二区三区| 一区二区三区你懂的| 午夜在线a亚洲v天堂网2018| 国产精品自拍在线| 久久久久99| 亚洲国产高清一区| 一区二区三区.www| 国产精品嫩草影院av蜜臀| 欧美伊人久久| 亚洲国产精品va在看黑人| 亚洲手机在线| 国产婷婷色一区二区三区在线| 久久xxxx| 亚洲麻豆一区| 欧美一区二区精品| 亚洲大片在线| 欧美日本亚洲| 欧美影院一区| 亚洲欧洲视频| 久久九九免费视频| 亚洲精品乱码久久久久久| 国产精品videosex极品| 久久精品国产99国产精品| 亚洲欧洲精品一区| 久久久www| 一区二区三区日韩精品视频| 国产老肥熟一区二区三区| 免费观看在线综合色| 亚洲一区视频在线| 欧美激情视频网站| 久久精品国产免费| 亚洲最新在线| 极品少妇一区二区| 国产精品美女www爽爽爽视频| 久久久国产精品亚洲一区| 一本色道久久综合亚洲精品小说| 久久亚洲欧美国产精品乐播| 亚洲一区二区三区色| 亚洲高清av| 国产一区二区三区高清在线观看| 欧美日韩色婷婷| 久热re这里精品视频在线6| 亚洲欧美日韩中文在线制服| 亚洲三级免费电影| 欧美成人a视频| 欧美一区二区三区另类| 一区二区三区 在线观看视| 在线免费日韩片| 国产一区二区av| 国产精品拍天天在线| 欧美日韩1区2区| 牛夜精品久久久久久久99黑人| 欧美亚洲视频在线观看| 9久re热视频在线精品| 亚洲国产欧美一区二区三区丁香婷| 久久狠狠一本精品综合网| 亚洲欧美999| 亚洲午夜羞羞片| 一本色道久久综合狠狠躁篇的优点 | 欧美a级理论片| 久久精品一区四区| 欧美一级二区| 性伦欧美刺激片在线观看| 亚洲素人一区二区| 99视频超级精品| 一本大道av伊人久久综合| 日韩视频在线一区二区| 亚洲欧洲日夜超级视频| 亚洲精品乱码久久久久| 亚洲日本理论电影| 亚洲免费电影在线观看| 亚洲精品国产日韩| 日韩亚洲欧美一区| 一级日韩一区在线观看| 一本色道久久综合精品竹菊 | 久久激情久久| 久久久蜜桃一区二区人| 老色批av在线精品| 米奇777在线欧美播放| 免费看亚洲片| 亚洲黄色大片| 一级成人国产| 亚洲欧美激情视频| 久久国产天堂福利天堂| 老鸭窝亚洲一区二区三区| 欧美国产欧美亚洲国产日韩mv天天看完整 | 国内精品美女在线观看| 黄色成人在线网址| 亚洲国产小视频| 在线一区二区三区四区| 欧美一级成年大片在线观看| 久久这里有精品视频| 欧美高清视频| 亚洲免费观看高清完整版在线观看熊| 在线亚洲精品福利网址导航| 欧美一二三区在线观看| 免费不卡在线视频| 欧美午夜精品久久久久久孕妇| 国产精品一区久久久| 在线成人亚洲| 亚洲午夜激情在线| 久久精品国产亚洲5555| 亚洲国产日韩欧美一区二区三区| 亚洲小视频在线| 久久久久一区二区三区四区| 欧美日韩国产一中文字不卡 | 国产日韩精品视频一区| 亚洲高清视频中文字幕| 亚洲自拍偷拍网址| 欧美h视频在线| 在线亚洲激情| 免费观看在线综合| 国产精品网站一区| 亚洲精品免费在线观看| 久久成人人人人精品欧| 亚洲国产精品va在线看黑人动漫| 亚洲一级在线| 欧美高清视频| 狠狠入ady亚洲精品经典电影| 日韩午夜电影av| 久久久久久综合| 一区二区三区回区在观看免费视频| 久久久噜噜噜久噜久久| 国产精品乱码一区二三区小蝌蚪| 亚洲国产精品第一区二区| 欧美一区观看| 一本久道久久综合狠狠爱| 狂野欧美一区| 国产真实乱偷精品视频免| 亚洲综合日韩在线| 亚洲观看高清完整版在线观看| 午夜精品国产更新| 欧美午夜不卡| 日韩午夜电影av| 欧美高清在线视频| 久久免费高清| 国产视频在线观看一区二区| 午夜精品久久久久久久久久久久| 91久久精品久久国产性色也91| 久久久久国产精品www | 国产一区二区三区四区在线观看 | 麻豆久久婷婷| 亚洲第一精品福利| 久久性色av| 久久国产精品黑丝| 国产亚洲电影| 欧美中在线观看| 亚洲欧美日产图| 国产精品一国产精品k频道56| 亚洲视频综合| 一区二区三区四区五区视频| 欧美日韩精品伦理作品在线免费观看| 亚洲激情在线观看| 欧美激情亚洲另类| 欧美大片91| 99v久久综合狠狠综合久久| 亚洲国产免费| 欧美日韩在线播放三区| 亚洲一区二区免费在线| 夜夜嗨一区二区| 国产欧美日本一区视频| 欧美在线观看www| 久久不射中文字幕| 亚洲电影激情视频网站|