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

            那誰的技術博客

            感興趣領域:高性能服務器編程,存儲,算法,Linux內核
            隨筆 - 210, 文章 - 0, 評論 - 1183, 引用 - 0
            數據加載中……

            探索C++的秘密之二:重載,覆蓋,和隱藏

            這幾個概念都有一個共同點:函數名稱相同,所以不免讓人混淆,大致的區別如下:

            重載(overload):
            必須在一個域中,函數名稱相同但是函數參數不同,重載的作用就是同一個函數有不同的行為,因此不是在一個域中的函數是無法構成重載的,這個是重載的重要特征

            覆蓋(override):
            覆蓋指的是派生類的虛擬函數覆蓋了基類的同名且參數相同的函數,既然是和虛擬函數掛鉤,說明了這個是一個多態支持的特性,所謂的覆蓋指的是用基類對象的指針或者引用時訪問虛擬函數的時候會根據實際的類型決定所調用的函數,因此此時派生類的成員函數可以"覆蓋"掉基類的成員函數.
            注意唯有同名且參數相同還有帶有virtual關鍵字并且分別在派生類和基類的函數才能構成虛擬函數,這個也是派生類的重要特征.
            而且,由于是和多態掛鉤的,所以只有在使用類對象指針或者引用的時候才能使用上.
            總之一句話:覆蓋函數都是虛函數,反之不然~~

            隱藏(hide):
            指的是派生類的成員函數隱藏了基類函數的成員函數.隱藏一詞可以這么理解:在調用一個類的成員函數的時候,編譯器會沿著類的繼承鏈逐級的向上查找函數的定義,如果找到了那么就停止查找了,所以如果一個派生類和一個基類都有同一個同名(暫且不論參數是否相同)的函數,而編譯器最終選擇了在派生類中的函數,那么我們就說這個派生類的成員函數"隱藏"了基類的成員函數,也就是說它阻止了編譯器繼續向上查找函數的定義....
            回到隱藏的定義中,前面已經說了有virtual關鍵字并且分別位于派生類和基類的同名,同參數函數構成覆蓋的關系,因此隱藏的關系只有如下的可能:
            1)必須分別位于派生類和基類中
            2)必須同名
            3)參數不同的時候本身已經不構成覆蓋關系了,所以此時是否是virtual函數已經不重要了
            ??當參數相同的時候就要看時候有virtual關鍵字了,有的話就是覆蓋關系,沒有的時候就是隱藏關系了

            上面的解說大體把三者的區別給說清楚了,但是還有一些疑惑的地方,以下以代碼例子說明.

            很多人分辨不清隱藏和覆蓋的區別,因為他們都是發生在基類和派生類之中的.但是它們之間最為重要的區別就是:
            覆蓋的函數是多態的,是存在于vtbl之中的函數才能構成"覆蓋"的關系,而隱藏的函數都是一般的函數,不支持多態,在編譯階段就已經確定下來了.


            class ?Base
            {
            public :
            virtual ? void ?f( float ?x) {cout << " Base::f(folat) " << x << endl;}
            ????????
            void ?g( float ?x) {cout << " Base::g(float) " << x << endl;} ???
            }
            ;

            class ?Derived: public ?Base
            {
            public ?:
            ????
            virtual ? void ?f( float ?x) {cout << " Derived::f(float) " << x << endl;}
            ????????????
            void ?g( int ?x) {cout << " Deriver::g(int) " << x << endl;}
            }
            ;

            int ?main()
            {
            ????Derived?d;
            ????Base?
            * pb =& d;
            ????Derived?
            * pd =& d;
            ????pb
            -> f( 3.14f );
            ????pd
            -> f( 3.14f );
            ????pb
            -> g( 3.14f );??? // 輸出結果:Base::g(float)3.14
            ????pd -> g( 3.14f );??? // 輸出結果:Dervied::g(int)3
            ???? return ? 0 ;
            }



            在調用f函數的時候,派生類Derived的f函數覆蓋了基類Base的f函數,而派生類Derived的g函數隱藏了基類Base的g函數.
            為什么?理由很簡單,f函數是virtual函數,但是g函數不是.我們可以把Base類和Derived類看成這樣的一個struct:
            struct?Base
            {
            ????
            void??????????(*g)(float);??//?Base類型的函數指針,不可變
            ????struct?VTABLE??*__vptr;?????//?虛擬函數指針數組,可變
            }
            ;

            void?__Baseg(float)
            {
            ????cout
            <<"Base::g(folat)"<<x<<endl;
            }


            struct?Derived
            {
            ????
            void??????????(*g)(float);??//?Derived類型的函數指針,不可變
            ????struct?VTABLE??*__vptr;?????//?虛擬函數指針數組,可變
            }
            ;

            void?__Derivedg(float)
            {
            ????cout
            <<"Deriver::g(int)"<<x<<endl;
            }


            struct?VTABLE
            {
            ????
            void??????????(*f)(float);??//?函數指針
            }
            ;

            void?__Basef(float)
            {
            ????cout
            <<"Base::f(folat)"<<x<<endl;
            }


            void?__Derivedf(float)
            {
            ????cout
            <<"Deriver::f(int)"<<x<<endl;
            }



            在程序編譯的時候,函數指針f就已經是確定的了,但是__vptr根據不同的而有分別,而這個變化是運行期動態決定的.
            也就是說:f的地址不可變,__vptr可變.
            回到上面的例子中,Base *pb=&d;的時候只是用Derived類對象d的__vptr修改了Base類pb的__vptr指針,但是當Base類成員建立的
            時候f函數指針就是不能改變的.

            當函數被聲明為virtual的時候,就激活了多態機制,程序在運行的時候會根據類型的實際類型到VTABLE中查找函數指針,因此對函數g的調用就是這樣子的:
            pb->__vptr->g();
            而對f的調用就是一般的類成員函數指針的調用了:pb->f(),因為這個類型在程序編譯的時候已經確認了,所以在程序運行的時候是不能發生改變的.

            綜上,可以把
            Derived d;
            Base *pb=&d;
            的過程分解為:
            d.g = __Derivedg;
            d.__vptr->f = __Derivedf;
            pb->g = __Baseg;? ?? ?? ?? ?// 這里根據指針的真正類型確定函數指針
            pb->__vptr = d.__vptr;? ?? ?// 這里只是簡單的指針賦值,因此訪問到的就是Derived的函數了
            最后在調用:
            pb->f(3.14f);
            pb->g(3.14f);
            實際上是:
            pb->__vptr->__Derivedf(3.14f);
            __Baseg(3.14f);
            這么寫就明白最后在調用的時候為什么會用那樣的結果了,可以看出多了一個__vptr這個間接層實現了所謂的"動態綁定".

            最后,需要說明的一點是:實際上在c++中,非static和非virtual的函數指針并不會在一個class中保存它的函數指針,上面把函數g的指針寫在struct里面只是為了方便說明這樣的問題:在編譯階段這個函數就已經是確定的不可改變的了.特此說明一下.

            posted on 2006-03-24 20:39 那誰 閱讀(3154) 評論(6)  編輯 收藏 引用 所屬分類: C\C++

            評論

            # re: 探索C++的秘密之二:重載,覆蓋,和隱藏   回復  更多評論   

            重載,重寫,和隱藏會不會更好理解一些。
            2006-03-24 21:12 | 沐楓

            # re: 探索C++的秘密之二:重載,覆蓋,和隱藏   回復  更多評論   

            我看到很多地方都是寫的覆蓋,這個講究的地方在哪里呢?兄臺給我講一下,謝啦~~
            2006-03-24 22:08 | 創系

            # re: 探索C++的秘密之二:重載,覆蓋,和隱藏   回復  更多評論   

            沒什么講究啦,就是看什么詞容易讓人理解,就是什么詞好。
            2006-03-27 13:27 | 沐楓

            # re: 探索C++的秘密之二:重載,覆蓋,和隱藏   回復  更多評論   

            寫的很好! 謝謝!
            2008-06-11 16:51 | anonymous

            # re: 探索C++的秘密之二:重載,覆蓋,和隱藏 [未登錄]  回復  更多評論   

            里面有很多不一致的地方哦
            2010-07-29 07:33 | haha

            # re: 探索C++的秘密之二:重載,覆蓋,和隱藏 [未登錄]  回復  更多評論   

            謝謝分享,我還一直沒弄明白,這下明白了。
            2011-12-20 19:36 | Cloud
            午夜久久久久久禁播电影| 亚洲国产成人久久一区WWW| 亚洲女久久久噜噜噜熟女| 性做久久久久久久| 东京热TOKYO综合久久精品| 国产福利电影一区二区三区久久久久成人精品综合 | 国产精品9999久久久久| 狠狠色综合网站久久久久久久| 久久www免费人成精品香蕉| 久久精品国产男包| 国产91久久精品一区二区| 香港aa三级久久三级老师2021国产三级精品三级在| 久久久久亚洲AV成人网人人网站| 九九精品久久久久久噜噜| 欧美伊人久久大香线蕉综合| 久久精品无码专区免费东京热| 欧美麻豆久久久久久中文| 亚洲精品乱码久久久久久久久久久久 | 青青国产成人久久91网| 奇米影视7777久久精品人人爽| 好属妞这里只有精品久久| 午夜精品久久久久久影视riav| 久久综合给合久久狠狠狠97色 | 人妻无码精品久久亚瑟影视| 伊人久久大香线蕉av不卡| 国产免费久久精品丫丫| 久久99国产乱子伦精品免费| 亚洲国产成人乱码精品女人久久久不卡| 久久久久亚洲精品天堂| 色狠狠久久综合网| 久久久WWW免费人成精品| 国产韩国精品一区二区三区久久| 久久综合亚洲色一区二区三区| 国产巨作麻豆欧美亚洲综合久久| 国产人久久人人人人爽| 亚洲精品无码久久久久久| 武侠古典久久婷婷狼人伊人| 久久综合亚洲色HEZYO国产| 久久久久女教师免费一区| 久久有码中文字幕| 国产真实乱对白精彩久久|