• <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>
            franksunny的個(gè)人技術(shù)空間
            獲得人生中的成功需要的專(zhuān)注與堅(jiān)持不懈多過(guò)天才與機(jī)會(huì)。 ——C.W. Wendte

            這次失業(yè)之后,突然發(fā)現(xiàn)現(xiàn)在工作好像真的不是很好找,沒(méi)辦法,主動(dòng)權(quán)不在自己手里,靜下心來(lái)想想就當(dāng)通過(guò)筆試來(lái)給自己查漏補(bǔ)缺吧,昨天筆試遇到一個(gè)虛擬繼承的概念,這不雖說(shuō)2分的題,但是這個(gè)玩意有大內(nèi)容,我學(xué)習(xí)了下,也就先整個(gè)入門(mén)出來(lái)吧:

             

            為什么要引入虛擬繼承?

            虛擬繼承在一般的應(yīng)用中很少用到,所以也往往被忽視,這也主要是因?yàn)樵?/span>C++中,多重繼承是不推薦的,也并不常用,而一旦離開(kāi)了多重繼承,虛擬繼承就完全失去了存在的必要(因?yàn)檫@樣只會(huì)降低效率和占用更多的空間,關(guān)于這一點(diǎn),我自己還沒(méi)有太多深刻的理解,有興趣的可以看網(wǎng)絡(luò)上白楊的作品RTTI、虛函數(shù)和虛類(lèi)的開(kāi)銷(xiāo)分析及使用指導(dǎo),說(shuō)實(shí)話我目前還沒(méi)看得很明白,高人可以指點(diǎn)下我)。

            以下面的一個(gè)例子為例:

            #include <iostream.h>

            #include <memory.h>

            class CA

            {

                int k; //如果基類(lèi)沒(méi)有數(shù)據(jù)成員,則在這里多重繼承編譯不會(huì)出現(xiàn)二義性

            public:

                void f() {cout << "CA::f" << endl;}

            };

             

            class CB : public CA

            {

            };

             

            class CC : public CA

            {

            };

             

            class CD : public CB, public CC

            {

            };

             

            void main()

            {

                CD d;

                d.f();

            }

            當(dāng)編譯上述代碼時(shí),我們會(huì)收到如下的錯(cuò)誤提示:

            error C2385: 'CD::f' is ambiguous

            即編譯器無(wú)法確定你在d.f()中要調(diào)用的函數(shù)f到底是哪一個(gè)。這里可能會(huì)讓人覺(jué)得有些奇怪,命名只定義了一個(gè)CA::f,既然大家都派生自CA,那自然就是調(diào)用的CA::f,為什么還無(wú)法確定呢?

            這是因?yàn)?span style="COLOR: red">編譯器在進(jìn)行編譯的時(shí)候,需要確定子類(lèi)的函數(shù)定義,如CA::f是確定的,那么在編譯CBCC時(shí)還需要在編譯器的語(yǔ)法樹(shù)中生成CB::fCC::f等標(biāo)識(shí),那么,在編譯CD的時(shí)候,由于CBCC都有一個(gè)函數(shù)f,此時(shí),編譯器將試圖生成這兩個(gè)CD::f標(biāo)識(shí),顯然這時(shí)就要報(bào)錯(cuò)了。(當(dāng)我們不使用CD::f的時(shí)候,以上標(biāo)識(shí)都不會(huì)生成,所以,如果去掉d.f()一句,程序?qū)㈨樌ㄟ^(guò)編譯

             

            要解決這個(gè)問(wèn)題,有兩個(gè)方法:

            1、重載函數(shù)f():此時(shí)由于我們明確定義了CD::f,編譯器檢查到CD::f()調(diào)用時(shí)就無(wú)需再像上面一樣去逐級(jí)生成CD::f標(biāo)識(shí)了;

            此時(shí)CD的元素結(jié)構(gòu)如下:

            |CB(CA)|

            |CC(CA)|

            故此時(shí)的sizeof(CD) = 8;CBCC各有一個(gè)元素k

            2、使用虛擬繼承:虛擬繼承又稱(chēng)作共享繼承,這種共享其實(shí)也是編譯期間實(shí)現(xiàn)的,當(dāng)使用虛擬繼承時(shí),上面的程序?qū)⒆兂上旅娴男问剑?/span>

            #include <iostream.h>

            #include <memory.h>

            class CA

            {

                int k;

            public:

                void f() {cout << "CA::f" << endl;}

            };

             

            class CB : virtual public CA  //也有一種寫(xiě)法是class CB : public virtual CA

            {                       //實(shí)際上這兩種方法都可以

            };

             

            class CC : virtual public CA

            {

            };

             

            class CD : public CB, public CC

            {

            };

             

            void main()

            {

                CD d;

                d.f();

            }

            此時(shí),當(dāng)編譯器確定d.f()調(diào)用的具體含義時(shí),將生成如下的CD結(jié)構(gòu):

            |CB|

            |CC|

            |CA|

            同時(shí),在CBCC中都分別包含了一個(gè)指向CA的虛基類(lèi)指針列表vbptrvirtual base table pointer),其中記錄的是從CBCC的元素到CA的元素之間的偏移量。此時(shí),不會(huì)生成各子類(lèi)的函數(shù)f標(biāo)識(shí),除非子類(lèi)重載了該函數(shù),從而達(dá)到“共享”的目的(這里的具體內(nèi)存布局,可以參看鉆石型繼承內(nèi)存布局,在白楊的那篇文章中也有)。

            也正因此,此時(shí)的sizeof(CD) = 12(兩個(gè)vbptr + sizoef(int);

             

            另注:

            如果CBCC中各定義一個(gè)int型變量,則sizeof(CD)就變成20(兩個(gè)vbptr + 3個(gè)sizoef(int)

            如果CA中添加一個(gè)virtual void f1(){}sizeof(CD) = 16(兩個(gè)vbptr + sizoef(int)+vptr;

            再添加virtual void f2(){}sizeof(CD) = 16不變。原因如下所示:帶有虛函數(shù)的類(lèi),其內(nèi)存布局上包含一個(gè)指向虛函數(shù)列表的指針(vptr),這跟有幾個(gè)虛函數(shù)無(wú)關(guān)。

            以上內(nèi)容涉及到類(lèi)對(duì)象內(nèi)存布局問(wèn)題,本人還難以做過(guò)多展開(kāi),先貼這么多,本篇文章只是考慮對(duì)于虛擬繼承進(jìn)行入門(mén),至于效率、應(yīng)用等未作展開(kāi)。本文在網(wǎng)上文章基礎(chǔ)上修改了下而得此篇,原文載于http://blog.csdn.net/billdavid/archive/2004/06/23/24317.aspx

            另外關(guān)于虛繼承和虛基類(lèi)的討論,博客園有篇文章《虛繼承與虛基類(lèi)的本質(zhì)》,總結(jié)得更為詳細(xì)一點(diǎn)。

             

             

            posted on 2008-10-16 16:55 frank.sunny 閱讀(13781) 評(píng)論(8)  編輯 收藏 引用 所屬分類(lèi): C/C++學(xué)習(xí)和實(shí)踐

            FeedBack:
            # re: [整理]虛擬繼承入門(mén)
            2008-10-16 17:46 | 浪跡天涯
            學(xué)習(xí)了...不斷積累!  回復(fù)  更多評(píng)論
              
            # re: [整理]虛擬繼承入門(mén)
            2008-10-16 19:23 | giscn
            最好不要出現(xiàn)必須用虛擬繼承的情況  回復(fù)  更多評(píng)論
              
            # re: [整理]虛擬繼承入門(mén)
            2008-10-16 22:28 | frank.sunny
            @giscn
            的確,這個(gè)技術(shù)雖然從OO角度來(lái)說(shuō)是存在的,但是我們的確應(yīng)該避免,說(shuō)真的,工作了這么久也沒(méi)有接觸過(guò),昨天筆試遇到了這么個(gè)概念,才想起翻出來(lái)看看,而且要搞透它涉及的OO概念需要很全面很扎實(shí),我也就在別人基礎(chǔ)上入入門(mén)而已  回復(fù)  更多評(píng)論
              
            # re: [整理]虛擬繼承入門(mén)
            2008-10-17 17:49 | PDF閱讀器下載
            這個(gè)技術(shù)接觸不多,打算自己好好看一下  回復(fù)  更多評(píng)論
              
            # re: [整理]虛擬繼承入門(mén)
            2009-09-24 16:45 | 小人物做大事
            在COM技術(shù)中需要掌握C++的這一特性,有興趣的可以去研究一下COM編程,進(jìn)一步的體會(huì)一下多重繼承的應(yīng)用。  回復(fù)  更多評(píng)論
              
            # re: [整理]虛擬繼承入門(mén)
            2009-11-01 21:04 | Dim
            這個(gè)確實(shí)很少用到哦~~~
            在學(xué)校就更少用到了……
            感覺(jué),就目前自己的應(yīng)用范圍來(lái)說(shuō)僅能當(dāng)理論學(xué)習(xí)學(xué)習(xí)……  回復(fù)  更多評(píng)論
              
            # re: [整理]虛擬繼承入門(mén)
            2010-08-23 09:51 | touseit
            @Dim
            log4cplus 好像用了不少虛擬繼承的。一般涉及到菱形繼承的時(shí)候就要注意了,這時(shí)就要考慮到虛擬繼承了。  回復(fù)  更多評(píng)論
              
            # re: [整理]虛擬繼承入門(mén)
            2010-09-27 11:14 | BK
            ??也正因此,此時(shí)的sizeof(CD) = 12(兩個(gè)vbptr + sizoef(int));
            這句有問(wèn)題,是兩個(gè)sizeof(int)+一個(gè)vbptr吧?
              回復(fù)  更多評(píng)論
              

            常用鏈接

            留言簿(13)

            隨筆分類(lèi)

            個(gè)人其它博客

            基礎(chǔ)知識(shí)鏈接

            最新評(píng)論

            閱讀排行榜

            評(píng)論排行榜

            久久久www免费人成精品| 久久亚洲国产中v天仙www| 久久成人18免费网站| 久久久噜噜噜www成人网| 狠狠综合久久AV一区二区三区| 日本久久中文字幕| 色老头网站久久网| 亚洲国产成人久久笫一页 | 伊人久久大香线蕉亚洲| 99久久这里只精品国产免费| 热久久视久久精品18| 亚洲国产精品18久久久久久| 人妻无码久久一区二区三区免费| 欧洲成人午夜精品无码区久久| 国产产无码乱码精品久久鸭| 成人免费网站久久久| 国产精品午夜久久| 中文字幕无码久久人妻| 国产成人无码精品久久久性色| 国产精品美女久久久久| 久久久久久狠狠丁香| 热RE99久久精品国产66热| 久久久久久久97| 久久精品国产99国产电影网| 久久久久久噜噜精品免费直播| 日韩人妻无码精品久久免费一| 狠狠色丁香婷婷综合久久来 | 国产一级做a爰片久久毛片| 国产三级观看久久| 亚洲中文字幕无码久久2020| 婷婷久久综合九色综合98| 日本加勒比久久精品| 1000部精品久久久久久久久| 伊人热热久久原色播放www| 国产精品99久久99久久久| 久久久久无码精品国产app| 九九精品99久久久香蕉| 久久天天婷婷五月俺也去| 91久久香蕉国产熟女线看| 久久国产免费观看精品3| 一本久久免费视频|