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

            搜索

            •  

            積分與排名

            • 積分 - 175914
            • 排名 - 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 閱讀(1923) 評論(0)  編輯 收藏 引用 所屬分類: C & C++
            国产精品欧美亚洲韩国日本久久| 久久人妻AV中文字幕| 精品久久8x国产免费观看| 久久亚洲日韩精品一区二区三区| 精品久久久中文字幕人妻 | 狠狠色丁香久久婷婷综合蜜芽五月| 久久久久国产精品嫩草影院| 亚洲精品综合久久| 久久久精品人妻一区二区三区蜜桃| 亚洲精品乱码久久久久久久久久久久 | 久久精品亚洲中文字幕无码麻豆| 99久久人妻无码精品系列| 久久婷婷五月综合97色直播| 噜噜噜色噜噜噜久久| 久久久久99精品成人片试看 | 要久久爱在线免费观看| 久久精品蜜芽亚洲国产AV| 中文精品久久久久国产网址| 亚洲精品无码久久不卡| 国产精品美女久久久久网| 一本久久精品一区二区| 嫩草影院久久国产精品| 亚洲人成精品久久久久| 久久国产午夜精品一区二区三区| 麻豆亚洲AV永久无码精品久久| 久久国产午夜精品一区二区三区| 久久综合综合久久综合| 亚洲国产成人久久一区久久| 国产精品天天影视久久综合网| 成人久久免费网站| 香蕉久久久久久狠狠色| 久久久久久青草大香综合精品| 72种姿势欧美久久久久大黄蕉| 久久久精品国产免大香伊| 久久午夜福利电影| 久久国产精品免费| 久久九九久精品国产免费直播| 亚洲国产精品一区二区久久| 99久久夜色精品国产网站| 久久青青草原综合伊人| 久久九九青青国产精品|