• <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>

            5D空間

            學習總結(jié)與經(jīng)驗交流

               :: 首頁 :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理 ::
              在學習多重繼承、二義性、虛基類的時候遇到了一些困惑。經(jīng)過一定的學習摸索,雖然在底層機制上還不太清楚,但是在抽象層面上有了一定理解。
              書上只有一個虛基類的概念,即在繼承的時候加上關鍵字virtual。這里我們姑且把這種繼承方式叫做虛繼承。現(xiàn)在先來說一下虛繼承和一般繼承的區(qū)別。
              要解釋這一系列問題,我們首先要搞清楚這一系列概念意味著什么。多重繼承不用贅述。現(xiàn)在先就二義性和虛繼承談談我的看法。

              在一般的繼承中(非虛繼承),每一個派生類都保存了一份完整的基類副本。考慮以下繼承:
            class A
            {
               
            void print();
            }
            ;

            class B : public A
            {
               
            void print();
            }
            ;

            class C : public C
            {
               
            void print();
            }
            ;

            在這樣一系列繼承體系中,A包含一份print(),B包含了兩份,而C則包含了三分prin()。這里總共有6份獨立的print()函數(shù)。雖然在C中調(diào)用B::print()感覺和B中調(diào)用print()效果一樣,但他們確實是作為兩個副本存在。  而在虛繼承中,考慮如下繼承:
            class A
            {
               
            void print();
            }
            ;

            class B : virtual public A
            {
               
            void print();
            }
            ;

            class C : virtual public C
            {
               
            void print();
            }
            ;
            B只含有一份print()副本,但是卻可以通過A::print()調(diào)用A的print()函數(shù)。同理,C也只包含了一份print()副本。這里總共只有3分print()副本。虛繼承中基類的數(shù)據(jù)并沒有變多一份給派生類,而只是使用權移交了,就好像A有一棟樓,虛繼承給B,名義上B也擁有了這棟樓,可以使用,但是并沒有真正為B另外建一棟一模一樣的樓。

              二義性:要解釋二義性,最好先定義一個概念:名字間隔。籠統(tǒng)地表達,一個名字的間隔就是某個數(shù)據(jù)的名字從繼承層次中首次出現(xiàn)到達最后派生類時中間隔了多少相同的名字。間隔越少,這個名字的優(yōu)先級越高。當然直接在最終類里面聲明的名字具有最高的優(yōu)先級。比如考慮一開始的普通繼承:
            class A
            {
               
            void print();
            }
            ;

            class B : public A
            {
               
            void print();
            }
            ;

            class C : public C
            {
               
            void print();
            }
            ;

            如果使用C的對象,那么A中的pirnt與C間隔最大,C中的print與C的間隔最短,所以如果直接調(diào)用C對象的print函數(shù),那么將調(diào)用C版本的print。如果C沒有定義一個print函數(shù),那么B中的print函數(shù)與C間隔最小,那么調(diào)用C對象的print函數(shù)時,將調(diào)用B版本的print函數(shù)。
              有了這個概念,現(xiàn)在來解釋二義性:如果存在兩個及其以上的名字距離最終派生類的距離最短(長度一樣),那么,根據(jù)剛才由名字間隔定義的優(yōu)先級別,在直接調(diào)用這個派生類對象的相應數(shù)據(jù)時,便不知道該調(diào)用哪個版本了(注意直接兩個字,因為可以通過二元::來分辨具體的版本以調(diào)用,所以即使名字存在二義性,如果未調(diào)用這些名字,編譯器可能不會報錯)。有兩種情況(到目前為止我看到的)可能導致二義性:1、在類中聲明了兩個名字一樣的成員:這是最糟糕的情況,因為如果這樣做了,沒有辦法彌補,但這也是最好的情況,因為編譯器根本不會讓你這么做。2、多繼承的時候繼承了兩個間隔一樣的名字:通常難以對付的是這種情況。
              關于上述第二種情況(多繼承),這些具有二義性的名字可能1、來自兩個基類各自的聲明,2、也可能來自兩個基類繼承自更高層次的同一基類(菱形繼承),3、也可能其中一個名字來自基類聲明,另一個名字來自另一個基類對更高層次基類的繼承。無論如何,只要同時存在兩個及其以上具有如果存在兩個及其以上的名字距離最終派生類的距離最短(長度一樣),那么就存在二義性。

            1、來自兩個基類各自聲明
            class B1
            {
               
            void print();
            }
            ;

            class B2 :
            {
               
            void print();
            }
            ;

            class C : public B1, public B2
            {
            }
            ;

            2、菱形繼承
            class A
            {
               
            void print();
            }
            ;

            class B1 : public A
            {
            }
            ;

            class B2 : public A
            {
            }
            ;

            class C : public B1, public B2
            {
            }
            ;

            3、其中一個名字來自基類聲明,另一個名字來自另一個基類對更高層次基類的繼承
            class A
            {
               
            void print();
            }
            ;

            class B1 : public A
            {
            }
            ;

            class B2 :
            {
               
            void print();   
            }
            ;

            class C : public B1, public B2
            {
            }
            ;
            (注意:雖然A版本的print是通過B1到達C的,但是A->B1->C的過程中,A版本的print與C之間并沒有間隔其他的print,這與B2版本的print一樣,所以他們具有相同的名字間隔,因此具有二義性)

              二義性的解決辦法:
              1、在最終派生類中定義一個相同名字的成員,這樣這個名字距離最終派生類最近,所以就會調(diào)用這個名字下的數(shù)據(jù)(通常教材里叫做這個名字把其他名字隱藏了)。這個名字(如果是函數(shù))你可以自己定義新的方法,也可以通過二元::調(diào)用你已知的存在二義性的名字中的某一個(注意:如果你選擇的調(diào)用版本不是該派生類的直接基類,那么該如何調(diào)用呢?比如A->B->C,那么從C的對象c調(diào)用A的print函數(shù),c.A::print()是否可行?我在vs2010上,雖然報錯但是編譯通過且正常運行。如果各位有任何見解或建議,希望不吝賜教。)
              2、使用虛繼承(針對菱形繼承等):回想一下虛繼承和普通繼承,通過虛繼承的方法可以消除重復副本帶來的二義性問題。比如在某一繼承層次上,這個某兩個名字具有二義性,然而順著繼承層次向上分析,卻發(fā)現(xiàn)這兩個名字其實是同一個東西的兩個副本,這個時候如果使用虛繼承,那么就使得這兩個副本變?yōu)橐粋€副本(準確地說,兩個副本都沒有了,因為只存在他們公共基類的那份數(shù)據(jù),虛基類得到的不過是使用權)。

            寫在后面:
              注意虛函數(shù)和虛繼承的區(qū)別:虛函數(shù)并沒有減少任何數(shù)據(jù)的存在,僅僅相當于在基類指針層面上建立了一種“調(diào)用最靠近對象類型的函數(shù)”的機制。然而虛繼承則是一種類的繼承方式,即,只創(chuàng)建派生類特有部分的數(shù)據(jù),繼承的數(shù)據(jù)按需從基類索取。所以雖然他們都是用virtual關鍵字,但似乎意思上聯(lián)系不大。
              另外,是用虛繼承能夠解決的問題相當有限。而且虛繼承面臨一個開銷問題,雖然從繼承層面上看,這是一個消除二義性的好方法,而且似乎對編程沒有什么副作用。這個道理與虛函數(shù)帶來的好處與開銷權衡問題差不多。一些書希望把這個問題留個程序員自己權衡,一些書則建議一律使用虛函數(shù)。不過應該指出,現(xiàn)在硬件設備能力的提升速度似乎在不斷削弱我們對開銷問題的顧忌(只要算法上不存在問題),所以即使你不打算從現(xiàn)在開始就全盤使用虛函數(shù)以及虛繼承(而且對于一般的小程序,即使不斷加上這些關鍵字也會使人厭煩吧,況且有些類似乎一輩子也不會成為基類呢?),但是請至少保持這樣一個念頭,多一種打算,多一條路嘛。
            posted on 2011-04-04 17:24 今晚打老虎 閱讀(3458) 評論(2)  編輯 收藏 引用 所屬分類: 學習筆記

            評論

            # re: 多重繼承、二義性、虛基類(虛繼承)之我見 2012-05-29 18:18 自己繼承自己
            孩子,代碼打錯了。

            class C : public C
            自己繼承自己?  回復  更多評論
              

            # re: 多重繼承、二義性、虛基類(虛繼承)之我見 2012-07-20 13:14 CL
            可以啊,自慰.@自己繼承自己
              回復  更多評論
              

            伊人色综合久久天天网| 亚洲国产成人久久一区WWW| 久久有码中文字幕| 无码任你躁久久久久久久| 久久夜色精品国产噜噜亚洲a| 亚洲欧美伊人久久综合一区二区 | 久久无码专区国产精品发布| 久久99这里只有精品国产| 久久精品人妻中文系列| 亚洲国产精品一区二区久久| 久久久久亚洲av毛片大| 国产精品禁18久久久夂久| 欧美成a人片免费看久久| 色欲av伊人久久大香线蕉影院| 蜜臀av性久久久久蜜臀aⅴ麻豆| 午夜天堂精品久久久久| 777久久精品一区二区三区无码| 久久99国产精一区二区三区| 久久伊人精品青青草原日本| 久久久久中文字幕| 少妇人妻88久久中文字幕| 久久播电影网| 国产∨亚洲V天堂无码久久久| 99久久精品国产一区二区三区| 日韩十八禁一区二区久久| 久久亚洲国产精品一区二区| 亚洲女久久久噜噜噜熟女| 亚洲国产成人久久一区久久| 久久免费视频一区| 久久精品成人影院| 一本一道久久精品综合| 国产精品久久久久影院嫩草| 久久夜色精品国产噜噜麻豆 | 国产亚洲精品自在久久| 人妻少妇精品久久| 久久久久99精品成人片| 久久久精品久久久久久 | 久久精品国产亚洲av麻豆图片 | 久久精品国产亚洲AV无码麻豆| 久久国产免费直播| 国内精品久久久久久久久|