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

            Welcome to ErranLi's Blog!

              C++博客 :: 首頁 :: 聯系 :: 聚合  :: 管理
              106 Posts :: 1 Stories :: 97 Comments :: 0 Trackbacks

            常用鏈接

            留言簿(12)

            搜索

            •  

            積分與排名

            • 積分 - 175087
            • 排名 - 151

            最新評論

            閱讀排行榜

            稍微懂得點oop的人都知道重載,那是多態性的重要體現!可是在c++中你能分清成員函數的重載、覆蓋嗎?這個好像也不難,重載存在與同一個類中,而覆蓋存在于派生類于基類中!可是如果再加上隱藏呢?說實話,以前我從來沒有聽說過這個概念!也不知道自己曾經捏造的程序,出了多少問題!看看林銳在《高質量c++編程指南》中的解釋。


               
            成員函數的重載、覆蓋(override)與隱藏很容易混淆,C++程序員必須要搞清楚概念,否則錯誤將防不勝防。

             

             

            8.2.1 重載與覆蓋

                成員函數被重載的特征:

            1)相同的范圍(在同一個類中);

            2)函數名字相同;

            3)參數不同;

            4virtual關鍵字可有可無。

                覆蓋是指派生類函數覆蓋基類函數,特征是:

            1)不同的范圍(分別位于派生類與基類);

            2)函數名字相同;

            3)參數相同;

            4)基類函數必須有virtual關鍵字。

                示例8-2-1中,函數Base::f(int)Base::f(float)相互重載,而Base::g(void)Derived::g(void)覆蓋。

             

             

            #include <iostream.h>

                class Base

            {

            public:

                         void f(int x){ cout << "Base::f(int) " << x << endl; }

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

                 virtual void g(void){ cout << "Base::g(void)" << endl;}

            };

             

             

                class Derived : public Base

            {

            public:

                 virtual void g(void){ cout << "Derived::g(void)" << endl;}

            };

             

             

                void main(void)

                {

                 Derived d;

                 Base *pb = &d;

                 pb->f(42);        // Base::f(int) 42

                 pb->f(3.14f);     // Base::f(float) 3.14

                 pb->g();          // Derived::g(void)

            }

            示例8-2-1成員函數的重載和覆蓋

               

            8.2.2 令人迷惑的隱藏規則

                本來僅僅區別重載與覆蓋并不算困難,但是C++的隱藏規則使問題復雜性陡然增加。這里“隱藏”是指派生類的函數屏蔽了與其同名的基類函數,規則如下:

            1)如果派生類的函數與基類的函數同名,但是參數不同。此時,不論有無virtual關鍵字,基類的函數將被隱藏(注意別與重載混淆)。

            2)如果派生類的函數與基類的函數同名,并且參數也相同,但是基類函數沒有virtual關鍵字。此時,基類的函數被隱藏(注意別與覆蓋混淆)。

                示例程序8-2-2a)中:

            1)函數Derived::f(float)覆蓋了Base::f(float)

            2)函數Derived::g(int)隱藏了Base::g(float),而不是重載。

            3)函數Derived::h(float)隱藏了Base::h(float),而不是覆蓋。

             

             

            #include <iostream.h>

                class Base

            {

            public:

                virtual void f(float x){ cout << "Base::f(float) " << x << endl; }

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

                        void h(float x){ cout << "Base::h(float) " << x << endl; }

            };

                class Derived : public Base

            {

            public:

                virtual void f(float x){ cout << "Derived::f(float) " << x << endl; }

            void g(int x){ cout << "Derived::g(int) " << x << endl; }

                        void h(float x){ cout << "Derived::h(float) " << x << endl; }

            };

            示例8-2-2a)成員函數的重載、覆蓋和隱藏

             

             

                據作者考察,很多C++程序員沒有意識到有“隱藏”這回事。由于認識不夠深刻,“隱藏”的發生可謂神出鬼沒,常常產生令人迷惑的結果。

            示例8-2-2b)中,pbpd指向同一地址,按理說運行結果應該是相同的,可事實并非這樣。

             

             

            void main(void)

            {

            Derived d;

            Base *pb = &d;

            Derived *pd = &d;

            // Good : behavior depends solely on type of the object

            pb->f(3.14f); // Derived::f(float) 3.14

            pd->f(3.14f); // Derived::f(float) 3.14

             

             

            // Bad : behavior depends on type of the pointer

            pb->g(3.14f); // Base::g(float) 3.14

            pd->g(3.14f); // Derived::g(int) 3        (surprise!)

             

             

            // Bad : behavior depends on type of the pointer

            pb->h(3.14f); // Base::h(float) 3.14      (surprise!)

            pd->h(3.14f); // Derived::h(float) 3.14

            }

            示例8-2-2b) 重載、覆蓋和隱藏的比較

            8.2.3 擺脫隱藏

                隱藏規則引起了不少麻煩。示例8-2-3程序中,語句pd->f(10)的本意是想調用函數Base::f(int),但是Base::f(int)不幸被Derived::f(char *)隱藏了。由于數字10不能被隱式地轉化為字符串,所以在編譯時出錯。

             

             

            class Base

            {

            public:

            void f(int x);

            };

            class Derived : public Base

            {

            public:

            void f(char *str);

            };

            void Test(void)

            {

            Derived *pd = new Derived;

            pd->f(10);    // error

            }

            示例8-2-3 由于隱藏而導致錯誤

             

             

                從示例8-2-3看來,隱藏規則似乎很愚蠢。但是隱藏規則至少有兩個存在的理由:

            u       寫語句pd->f(10)的人可能真的想調用Derived::f(char *)函數,只是他誤將參數寫錯了。有了隱藏規則,編譯器就可以明確指出錯誤,這未必不是好事。否則,編譯器會靜悄悄地將錯就錯,程序員將很難發現這個錯誤,流下禍根。

            u       假如類Derived有多個基類(多重繼承),有時搞不清楚哪些基類定義了函數f。如果沒有隱藏規則,那么pd->f(10)可能會調用一個出乎意料的基類函數f。盡管隱藏規則看起來不怎么有道理,但它的確能消滅這些意外。

             

             

            示例8-2-3中,如果語句pd->f(10)一定要調用函數Base::f(int),那么將類Derived修改為如下即可。

            class Derived : public Base

            {

            public:

            void f(char *str);

            void f(int x) { Base::f(x); }

            };

            posted on 2007-10-13 17:52 erran 閱讀(1917) 評論(0)  編輯 收藏 引用 所屬分類: C & C++
            亚洲va中文字幕无码久久不卡| 亚洲色欲久久久综合网东京热| 午夜不卡久久精品无码免费| 狠狠色丁香婷婷综合久久来来去| 久久精品aⅴ无码中文字字幕重口 久久精品a亚洲国产v高清不卡 | 久久久久亚洲精品男人的天堂| 免费国产99久久久香蕉| 99久久精品影院老鸭窝| 色综合久久久久综合体桃花网 | 国产美女亚洲精品久久久综合| 久久人人爽人爽人人爽av| 国产视频久久| 色综合久久中文字幕综合网| 久久本道久久综合伊人| 久久久久18| 一级a性色生活片久久无| 青青草原综合久久大伊人| 亚洲国产精品无码久久一线 | 久久美女人爽女人爽| 国产精品99久久久久久猫咪| 久久se精品一区二区影院| 亚洲第一永久AV网站久久精品男人的天堂AV | 久久伊人精品一区二区三区| 国产激情久久久久久熟女老人| 亚洲中文久久精品无码ww16| av国内精品久久久久影院| 久久精品国产99国产精品澳门| 国产精品免费久久久久电影网| 香港aa三级久久三级老师2021国产三级精品三级在 | 久久国产精品久久久| 国产精品欧美久久久久天天影视| 久久伊人精品青青草原日本| 久久强奷乱码老熟女网站| 99久久精品毛片免费播放| 久久久久国产一级毛片高清板| 久久福利资源国产精品999| 久久久久久人妻无码| 久久人人爽人人爽人人片AV麻豆| 亚洲中文字幕无码久久综合网| 久久精品www| 久久精品卫校国产小美女|