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

            重載(overload)、覆蓋(override)、隱藏(hide)的區(qū)別

            這個(gè)問題困擾我很久了,一直以為自己掌握了這個(gè)概念,關(guān)鍵是這個(gè)隱藏這個(gè)攪局的太麻煩,轉(zhuǎn)別人的博客關(guān)于這方面的說明,
            看了一下相對(duì)來說比較全面。http://blog.csdn.net/yanjun_1982/archive/2005/09/02/470405.aspx

                  這三個(gè)概念都是與OO中的多態(tài)有關(guān)系的。如果單是區(qū)別重載與覆蓋這兩個(gè)概念是比較容易的,但是隱藏這一概念卻使問題變得有點(diǎn)復(fù)雜了,下面說說它們的區(qū)別吧。

                   重載是指不同的函數(shù)使用相同的函數(shù)名,但是函數(shù)的參數(shù)個(gè)數(shù)或類型不同。調(diào)用的時(shí)候根據(jù)函數(shù)的參數(shù)來區(qū)別不同的函數(shù)。

                   覆蓋(也叫重寫)是指在派生類中重新對(duì)基類中的虛函數(shù)(注意是虛函數(shù))重新實(shí)現(xiàn)。即函數(shù)名和參數(shù)都一樣,只是函數(shù)的實(shí)現(xiàn)體不一樣。

                   隱藏是指派生類中的函數(shù)把基類中相同名字的函數(shù)屏蔽掉了。隱藏與另外兩個(gè)概念表面上看來很像,很難區(qū)分,其實(shí)他們的關(guān)鍵區(qū)別就是在多態(tài)的實(shí)現(xiàn)上。什么叫多態(tài)?簡單地說就是一個(gè)接口,多種實(shí)現(xiàn)吧。

                   還是引用一下別人的代碼來說明問題吧(引用自林銳的《高質(zhì)量C/C++編程指南》)。

            仔細(xì)看下面的代碼: 

            #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; } 

            };  

            看出什么了嗎?下面說明一下:

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

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

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

                   嗯,概念大概明白了,但是在實(shí)際的編程中,我們會(huì)因此遇到什么問題呢?再看下面的代碼:

            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  

            在第一種調(diào)用中,函數(shù)的行為取決于指針?biāo)赶虻膶?duì)象。在第二第三種調(diào)用中,函數(shù)的行為取決于指針的類型。所以說,隱藏破壞了面向?qū)ο缶幊讨卸鄳B(tài)這一特性,會(huì)使得OOP人員產(chǎn)生混亂。  

            不過隱藏也并不是一無是處,它可以幫助編程人員在編譯時(shí)期找出一些錯(cuò)誤的調(diào)用。但我覺得還是應(yīng)該盡量使用隱藏這一些特性,該加virtual時(shí)就加吧。


            c++ 重載 覆蓋 隱藏的區(qū)別和執(zhí)行方式

            成員函數(shù)被重載的特征
            (1)相同的范圍(在同一個(gè)類中);
            (2)函數(shù)名字相同;
            (3)參數(shù)不同;
            (4)virtual 關(guān)鍵字可有可無。
            覆蓋是指派生類函數(shù)覆蓋基類函數(shù),特征是
            (1)不同的范圍(分別位于派生類與基類);
            (2)函數(shù)名字相同;
            (3)參數(shù)相同;
            (4)基類函數(shù)必須有virtual 關(guān)鍵字。
            “隱藏”是指派生類的函數(shù)屏蔽了與其同名的基類函數(shù),規(guī)則如下
            (1)如果派生類的函數(shù)與基類的函數(shù)同名,但是參數(shù)不同。此時(shí),不論有無virtual關(guān)鍵字,基類的函數(shù)將被隱藏(注意別與重載混淆)。
            (2)如果派生類的函數(shù)與基類的函數(shù)同名,并且參數(shù)也相同,但是基類函數(shù)沒有virtual 關(guān)鍵字。此時(shí),基類的函數(shù)被隱藏(注意別與覆蓋混淆)
            3種情況怎么執(zhí)行:
            1。重載:看參數(shù)
            2。隱藏:用什么就調(diào)用什么
            3。覆蓋:調(diào)用派生類

             

            重載(overload),覆蓋(override),隱藏(hide)的區(qū)別  
            http://m.shnenglu.com/zgysx/archive/2007/03/12/19662.html 

            寫正題之前,先給出幾個(gè)關(guān)鍵字的中英文對(duì)照,重載(overload),覆蓋(override),隱藏(hide)。在早期的C++書籍中,可能翻譯的人不熟悉專業(yè)用語(也不能怪他們,他們不是搞計(jì)算機(jī)編程的,他們是英語專業(yè)的),常常把重載(overload)和覆蓋(override)搞錯(cuò)!

              我們先來看一些代碼及其編譯結(jié)果。

              實(shí)例一:
              #include "stdafx.h"
              #include <iostream.h>

              class CB
              {
              public:
                 
            void f(int)
                 {
                    cout << "CB::f(int)" << endl;
                 }
              };

              class CD : public CB
              {
              public:
                 
            void f(int,int)
                 {
                    cout << "CD::f(int,int)" << endl;
                 }
                 void test()
                 {
                   f(1);
                 }
              };

             int main(int argc, char* argv[])
             {
                return 0;
             }
            編譯了一下
            error C2660: 'f' : function does not take 1 parameters


            結(jié)論:在類CD這個(gè)域中,沒有f(int)這樣的函數(shù),基類中的void f(int)被隱藏

              如果把派生CD中成員函數(shù)void f(int,int)的聲明改成和基類中一樣,即f(int),基類中的void f(int)還是一樣被覆蓋,此時(shí)編譯不會(huì)出錯(cuò),在函數(shù)中test調(diào)用的是CD中的f(int) 

              所以,在基類中的某些函數(shù),如果沒有virtral關(guān)鍵字,函數(shù)名是f(參數(shù)是什么我們不管),那么如果在派生類CD中也聲明了某個(gè)f成員函數(shù),那么在類CD域中,基類中所有的那些f都被隱藏。
              如果你比較心急,想知道什么是隱藏,看文章最后的簡單說明,不過我建議你還是一步一步看下去。

              我們剛才說的是沒有virtual的情況,如果有virtual的情況呢??
              實(shí)例二:

            #include "stdafx.h"
            #include <iostream.h>

            class CB
            {
            public:
               
            virtual void f(int)
               {
                  cout << "CB::f(int)" << endl;
               }
            };

            class CD : public CB
            {
            public:
               
            void f(int)
               {
                  cout << "CD::f(int)" << endl;
               }
            };

            int main(int argc, char* argv[])
            {
              return 0;
            }

              這么寫當(dāng)然是沒問題了,在這里我不多費(fèi)口舌了,這是很簡單的,多態(tài),虛函數(shù),然后什么指向基類的指針指向派生類對(duì)象阿,通過引用調(diào)用虛函數(shù)阿什么的,屬性多的很咯,什么??你不明白??隨便找本C++的書,對(duì)會(huì)講多態(tài)和虛函數(shù)機(jī)制的哦!!
              這種情況我們叫覆蓋(override)!覆蓋指的是派生類的虛擬函數(shù)覆蓋了基類的同名且參數(shù)相同的函數(shù)!
              在這里,我要強(qiáng)調(diào)的是,這種覆蓋,要滿足兩個(gè)條件
             (a)有virtual關(guān)鍵字,在基類中函數(shù)聲明的時(shí)候加上就可以了
             (b)基類CB中的函數(shù)和派生類CD中的函數(shù)要一模一樣,什么叫一模一樣,函數(shù)名,參數(shù),返回類型三個(gè)條件。
              有人可能會(huì)對(duì)(b)中的說法質(zhì)疑,說返回類型也要一樣??
              是,覆蓋的話必須一樣,我試了試,如果在基類中,把f的聲明改成virtual int f(int),編譯出錯(cuò)了
              error C2555: 'CD::f' : overriding virtual function differs from 'CB::f' only by return type or calling convention
              所以,覆蓋的話,必須要滿足上述的(a)(b)條件

              那么如果基類CB中的函數(shù)f有關(guān)鍵字virtual ,但是參數(shù)和派生類CD中的函數(shù)f參數(shù)不一樣呢,
            實(shí)例三:
              #include "stdafx.h"
            #include <iostream.h>

            class CB
            {
             public:
                virtual  void f(int)
               {
                  cout << "CB::f(int)" << endl;
               }
            };


            class CD : public CB
            {
            public:
                void f(int,int)
               {
                 cout << "CD::f(int,int)" << endl;
               }

               void test()
               {
                  f(1);
               }
            };

            int main(int argc, char* argv[])
            {
             return 0;
            }

            編譯出錯(cuò)了,
             error C2660: 'f' : function does not take 1 parameters
              咦??好面熟的錯(cuò)??對(duì),和實(shí)例一中的情況一樣哦,結(jié)論也是基類中的函數(shù)被隱藏了。

              通過上面三個(gè)例子,得出一個(gè)簡單的結(jié)論
            如果基類中的函數(shù)和派生類中的兩個(gè)名字一樣的函數(shù)f
            滿足下面的兩個(gè)條件
            (a)在基類中函數(shù)聲明的時(shí)候有virtual關(guān)鍵字
            (b)基類CB中的函數(shù)和派生類CD中的函數(shù)一模一樣,函數(shù)名,參數(shù),返回類型都一樣。
            那么這就是叫做覆蓋(override),這也就是虛函數(shù),多態(tài)的性質(zhì)

            那么其他的情況呢??只要名字一樣,不滿足上面覆蓋的條件,就是隱藏了。

            下面我要講最關(guān)鍵的地方了,好多人認(rèn)為,基類CB中的f(int)會(huì)繼承下來和CD中的f(int,int)在派生類CD中構(gòu)成重載,就像實(shí)例一中想像的那樣。
              對(duì)嗎?我們先看重載的定義
              
            重載(overload):
              必須在一個(gè)域中,函數(shù)名稱相同但是函數(shù)參數(shù)不同,重載的作用就是同一個(gè)函數(shù)有不同的行為,因此不是在一個(gè)域中的函數(shù)是無法構(gòu)成重載的,這個(gè)是重載的重要特征
              必須在一個(gè)域中,而繼承明顯是在兩個(gè)類中了哦,所以上面的想法是不成立的,我們測試的結(jié)構(gòu)也是這樣,派生類中的f(int,int)把基類中的f(int)隱藏了
              所以,相同的函數(shù)名的函數(shù),在基類和派生類中的關(guān)系只能是覆蓋或者隱藏。

              在文章中,我把重載和覆蓋的定義都給了出來了,但是一直沒有給隱藏的定義,在最后,我把他給出來,這段話是網(wǎng)上google來的,比較長,你可以簡單的理解成,在派生類域中,看不到基類中的那個(gè)同名函數(shù)了,或者說,是并沒有繼承下來給你用,呵呵,如實(shí)例一 那樣。
              

            隱藏(hide):
            指的是派生類的成員函數(shù)隱藏了基類函數(shù)的成員函數(shù).隱藏一詞可以這么理解:在調(diào)用一個(gè)類的成員函數(shù)的時(shí)候,編譯器會(huì)沿著類的繼承鏈逐級(jí)的向上查找函數(shù)的定義,如果找到了那么就停止查找了,所以如果一個(gè)派生類和一個(gè)基類都有同一個(gè)同名(暫且不論參數(shù)是否相同)的函數(shù),而編譯器最終選擇了在派生類中的函數(shù),那么我們就說這個(gè)派生類的成員函數(shù)"隱藏"了基類的成員函數(shù),也就是說它阻止了編譯器繼續(xù)向上查找函數(shù)的定義.

            posted on 2011-12-25 15:10 sheng 閱讀(344) 評(píng)論(0)  編輯 收藏 引用


            只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


            導(dǎo)航

            <2011年12月>
            27282930123
            45678910
            11121314151617
            18192021222324
            25262728293031
            1234567

            統(tǒng)計(jì)

            常用鏈接

            留言簿(1)

            隨筆檔案

            收藏夾

            同行

            搜索

            最新評(píng)論

            閱讀排行榜

            評(píng)論排行榜

            无码任你躁久久久久久老妇| 久久久久久久99精品免费观看| 久久青青草原精品国产| 久久亚洲国产成人精品无码区| 国产精品一久久香蕉产线看| 亚洲乱码中文字幕久久孕妇黑人 | 性做久久久久久久久| 亚洲国产成人久久综合碰碰动漫3d | 久久精品毛片免费观看| 久久久久亚洲AV无码麻豆| 亚洲中文字幕久久精品无码喷水| 综合久久一区二区三区 | 青青青国产精品国产精品久久久久| 久久99亚洲网美利坚合众国| 国产成人综合久久综合 | 亚洲伊人久久大香线蕉苏妲己| 999久久久国产精品| 国产999精品久久久久久| 久久亚洲2019中文字幕| 国产69精品久久久久9999APGF| 久久久亚洲欧洲日产国码二区| AV无码久久久久不卡网站下载| 久久精品免费一区二区三区| 国产午夜精品理论片久久| 大香伊人久久精品一区二区| 国产色综合久久无码有码| 国产精品久久久久久影院| 久久本道综合久久伊人| 亚洲中文久久精品无码ww16| 99久久久精品| 亚洲午夜福利精品久久| 久久精品99久久香蕉国产色戒 | 国产精品青草久久久久福利99| 亚洲国产精品嫩草影院久久| 狠狠色丁香久久综合五月| 亚洲欧洲精品成人久久奇米网| 99国产欧美久久久精品蜜芽| 欧美日韩中文字幕久久久不卡| 韩国免费A级毛片久久| 久久久久这里只有精品| 久久午夜伦鲁片免费无码|