• <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++
            久久精品国产2020| 国产精品激情综合久久| 伊人色综合久久天天人手人婷| 久久久久18| 精品人妻久久久久久888| 2021精品国产综合久久| 久久久免费观成人影院| 狼狼综合久久久久综合网| 国产精品亚洲美女久久久| 久久精品一区二区三区AV| 91精品国产高清久久久久久91 | 囯产极品美女高潮无套久久久| 国产亚洲精久久久久久无码77777| 91久久精一区二区三区大全| 久久夜色精品国产| 久久国产精品99精品国产987| 亚洲一区精品伊人久久伊人| 国产99久久精品一区二区| 久久婷婷国产剧情内射白浆| 久久精品亚洲欧美日韩久久| 91精品国产色综久久| 国产精品99久久免费观看| 久久婷婷五月综合国产尤物app| 国产99久久久久久免费看| 久久超碰97人人做人人爱| 久久精品免费一区二区| 欧美精品九九99久久在观看| 丁香久久婷婷国产午夜视频| 久久国产精品一区二区| 2021久久国自产拍精品| 亚洲国产另类久久久精品黑人| 久久无码高潮喷水| 亚洲日韩中文无码久久| 久久中文字幕视频、最近更新| 久久久精品国产亚洲成人满18免费网站 | 性欧美丰满熟妇XXXX性久久久| 性做久久久久久久久浪潮| 亚洲欧美日韩久久精品| 色悠久久久久久久综合网| 亚洲第一永久AV网站久久精品男人的天堂AV | 久久久无码精品亚洲日韩按摩|