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

            zhgw01

            C++的重載與虛函數(shù)

            其實(shí)真正要說(shuō)的是虛函數(shù),不過(guò)其中要扯倒重載,所以順便也說(shuō)了下重載

            1. 重載
            1.1 簡(jiǎn)單重載
                  在C++中,是允許同名函數(shù)的存在
            int add(int i,int j);
            float add(float i,float);
                  而在c中,函數(shù)名是唯一的,所以為了區(qū)分int和float版本的add,你需要給它們起不同的名字,比如將int的命名為add_int,將float的命名為add_float,這樣做的壞處就是程序員要記住很多的函數(shù)名,雖然這些函數(shù)的功能是一樣的,而且也不直觀。
                  重載函數(shù)的存在,使得這種情況不在存在,它可以根據(jù)參數(shù)的類型,自動(dòng)調(diào)用合適的重載函數(shù),程序員只要記住要使用加法是調(diào)用add函數(shù)就可以,不同再像C中那樣猜測(cè)int版的add函數(shù)函數(shù)名是怎樣的,float版的函數(shù)名又是怎樣的。實(shí)際上,這兩個(gè)版本的add函數(shù)名在編譯后是不一樣的,編譯器自動(dòng)為它們進(jìn)行了修飾,比如int的修飾成add_int_int,float的修飾成add_float_float,不過(guò)這都只是編譯生成后的結(jié)果,對(duì)程序員來(lái)說(shuō),他只需知道add這個(gè)函數(shù),但他要調(diào)用的時(shí)候,比如使用了int參數(shù),編譯器根據(jù)參數(shù)推出該調(diào)用的版本,這里就是add_int,所有這一切程序員都是看不見(jiàn)的,也不需要關(guān)心,從而減輕了程序員的工作。
                  從上面的說(shuō)法也可以看出,函數(shù)重載也不是能亂重載的,重載的要求是:
                  1. 函數(shù)的參數(shù)類型不一樣,像上面的int和float的
                  2. 函數(shù)的參數(shù)個(gè)數(shù)不一樣
                  這是因?yàn)榫幾g器在修飾生成的函數(shù)名時(shí),一般用所有的參數(shù)類型來(lái)進(jìn)行修飾,比如void add(int,float) 修飾成add_int_float,void add(int,int,int) 修飾成add_int_int_int,這樣符合上面2條要求的重載函數(shù)最后生成的函數(shù)名是不一樣的。
                  可能有人會(huì)認(rèn)為,為什么不用返回值來(lái)區(qū)分,如果編譯器能推測(cè)出函數(shù)調(diào)用該返回什么值那自然沒(méi)什么問(wèn)題,但很多時(shí)候,往往只是調(diào)用函數(shù),使用函數(shù)的副作用,而不要求返回值,這個(gè)時(shí)候編譯器就推測(cè)不出了,比如
             
            void f();
            int f();

            int main()
            {
              f();
            }

            這個(gè)時(shí)候編譯器怎么知道調(diào)用哪個(gè)函數(shù)
            1.2 類中的重載函數(shù)
            不僅僅是全局函數(shù)可以重載,類中的函數(shù)也可以重載
            class Base
            {
            public:
               
            int f() const{
                  cout
            <<"Base:f()"<<endl;
                  
            return 1;
               }

               
               
            int f(stringconst{
                  
            return 1;
               }

            }
            ;

            這看起來(lái)跟全局的沒(méi)什么區(qū)別,但是當(dāng)涉及到繼承的時(shí)候,事情就變得麻煩起來(lái)
            class Derived1: public Base
            {
            public:
              
            //Redefinition
              int f() const{
                cout
            <<"Derived1:f()"<<endl;
                
            return 1;
              }

            }
            ;

            class Derived2: public Base
            {
            public:
              
            //change Return type
              void f()const
              
            {
                cout
            <<"Derived2:f()"<<endl;
               }

            }


            class Derived3: public Base
            {
            public:
              
            //change argument list
              int f(intconst
              
            {
                cout
            <<"Dervide3:f()"<<endl;
                
            return 1;
               }

            }

            子類中定義了跟父類同名的函數(shù),這個(gè)時(shí)候該如何辦?其實(shí)說(shuō)起來(lái)也很簡(jiǎn)單,只要子類定義了跟父類同名的函數(shù),不管是重寫了函數(shù)內(nèi)容(Dervied1),改變了返回類型(Derived2),還是改變了參數(shù)列表(Derived3),結(jié)果都一樣,子類中的同名函數(shù)將父類中的同名函數(shù)給隱藏了,只要子類中的函數(shù)是可見(jiàn)的,通過(guò)子類的對(duì)象調(diào)用父類的同名函數(shù)是不合法的,只能調(diào)用子類自身的同名函數(shù)。這就是所謂的名字隱藏。

            2. 重寫與虛函數(shù)
            2.1 基本知識(shí)
            虛函數(shù)在多態(tài)中經(jīng)常用到。你只要有一個(gè)基類的指針或引用,編譯器會(huì)為你調(diào)用該指針真正對(duì)應(yīng)的函數(shù)
            class Base
            {
            public;
              
            virtual void f()
              
            {
               cout
            <<"Base:f()"<<endl;
               }

            }
            ;

            class Derived:public Base
            {
            public:
               
            //You can also ingore virtual here
               virtual void f()
               

                 cout
            <<"Derived:f()"<<endl;
               }

            }
            ;

            int main()
            {
              Base
            * p=new Derived();
              p
            ->f();
              delete p;
            }


            程序輸出Derived::f(),這就是虛函數(shù)的作用。你可以不用關(guān)心基類指針到底指向那個(gè)子類,編譯器會(huì)為你調(diào)用正確的函數(shù)。
            這是因?yàn)榫幾g器使用了晚捆綁的緣故。
            當(dāng)一個(gè)類中有一個(gè)虛函數(shù)時(shí)(可以是因?yàn)樵陬愔新暶髁艘粋€(gè)虛函數(shù),也可以是因?yàn)榛愔杏刑摵瘮?shù),通過(guò)繼承得到)。編譯器就為這個(gè)類創(chuàng)造一個(gè)虛表(VTABLE),它當(dāng)中的虛函數(shù)位置是固定的,即使被繼承到子類中也一樣。當(dāng)定義了一個(gè)這個(gè)類的對(duì)象時(shí),編譯器會(huì)在這個(gè)對(duì)象中放入一個(gè)虛指針(VPTR)指向這個(gè)表。當(dāng)調(diào)用虛函數(shù)時(shí),編譯器在匯編代碼中插入
            一段代碼,這個(gè)代碼首先找到虛表,然后在通過(guò)偏移調(diào)用正確的函數(shù)。
            當(dāng)一個(gè)帶有虛函數(shù)的基類被繼承時(shí),這個(gè)VTABLE會(huì)被完整賦值,當(dāng)然對(duì)應(yīng)的函數(shù)地址會(huì)改成子類中的函數(shù)地址。如果子類另外聲明了虛函數(shù),就會(huì)在原來(lái)的虛函數(shù)后面添加上新的條目。
            重寫其實(shí)就是在子類中對(duì)父類的虛函數(shù)進(jìn)行重定義,因?yàn)橐话阕宇愑凶约旱奶匦浴?br>2.2 虛函數(shù)與重載
            如果子類中只是改寫了父類中虛函數(shù)的內(nèi)容,這就只是重寫(overriding),函數(shù)前面的virtual可以忽略掉
            但如果子類中改變了父類中虛函數(shù)的參數(shù)類型或個(gè)數(shù),那么父類中的同名函數(shù)就會(huì)被隱藏掉,這同普通的重載一樣,有一點(diǎn)不一樣的是,不可以通過(guò)改變返回類型來(lái)隱藏父類中的同名函數(shù)。
            2.3 切片
            當(dāng)用子類對(duì)象來(lái)初始化父類時(shí)(如函數(shù)中的call by value),新生成的父類對(duì)象會(huì)正確初始化它自身的vtable,而不會(huì)使用子類的vtable。

            注:
            雖然通過(guò)基類指針調(diào)用虛函數(shù),最后調(diào)用的是子類的函數(shù),但是如果使用的確實(shí)基類的默認(rèn)參數(shù)
            class Base
            {
            public:
              
            virtual void f(int i=0)
              
            {
                cout
            <<i<<endl;
               }

            }
            ;

            class Derived: public Base
            {
            public:
              
            virtual void f(int i=1)
              
            {
                cout
            <<i<<endl;
              }

            }
            ;


            int main()
            {
              Base
            * p=new Derived();
              p
            ->f(); //輸出的是0
            }

              

             
            注2:
            發(fā)生在private繼承時(shí)的問(wèn)題,父類中的虛函數(shù)是private的,當(dāng)它被private繼承時(shí),子類是無(wú)法訪問(wèn)到這個(gè)函數(shù)的,不過(guò)子類仍然可以override這個(gè)函數(shù)
            class Base
            {
            public:
              
            void nvi()
              
            {
                 vfun();
              }

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

            }
            ;

            class Derived1:private Base
            {
             
            public:
               
            void df()
               
            {
                 nvi(); 
            //調(diào)用base的nvi,由于這里沒(méi)有override vfun,所以輸出的是Base:vfun()
               }


            //事實(shí)上,這里不能直接調(diào)用base中的vfun,因?yàn)樗莗rivate繼承來(lái)的
            }


            class Derived2:private Base
            {
            public:
              
            void df()
              
            {
                nvi();
            //調(diào)用了Dervied2的vfun
               }

            private:
              
            void vfun();//要想override,必須重新聲明
            }
            ;

            void Derived2::vfun()
            {
              cout
            <<"Derived2::vfun()"<<endl;
            }

            posted on 2008-10-17 14:45 apacs 閱讀(5513) 評(píng)論(1)  編輯 收藏 引用 所屬分類: c++

            Feedback

            # re: C++的重載與虛函數(shù) 2012-07-26 11:54 曾經(jīng)半夏

            您這里說(shuō)的 名字隱藏:
            子類中定義了跟父類同名的函數(shù),這個(gè)時(shí)候該如何辦?其實(shí)說(shuō)起來(lái)也很簡(jiǎn)單,只要子類定義了跟父類同名的函數(shù),不管是重寫了函數(shù)內(nèi)容(Dervied1),改變了返回類型(Derived2),還是改變了參數(shù)列表(Derived3),結(jié)果都一樣,子類中的同名函數(shù)將父類中的同名函數(shù)給隱藏了,只要子類中的函數(shù)是可見(jiàn)的,通過(guò)子類的對(duì)象調(diào)用父類的同名函數(shù)是不合法的,只能調(diào)用子類自身的同名函數(shù)。這就是所謂的名字隱藏。
            和后面虛函數(shù)與重載中所說(shuō)的:
            但如果子類中改變了父類中虛函數(shù)的參數(shù)類型或個(gè)數(shù),那么父類中的同名函數(shù)就會(huì)被隱藏掉,這同普通的重載一樣,有一點(diǎn)不一樣的是,不可以通過(guò)改變返回類型來(lái)隱藏父類中的同名函數(shù)。
            應(yīng)該不是一個(gè)概念吧?
            因?yàn)槲疫@里可以通過(guò)直接引用父類的虛函數(shù)f()來(lái)調(diào)用,沒(méi)有在子類隱藏掉父類的函數(shù)。
            class Base
            {
            public:
            virtual void f()
            {
            cout<<"Base:f()"<<endl;
            }
            };

            class Derived:public Base
            {
            public:

            void f(int i=0)
            {
            cout<<"Derived:f()"<<endl;
            }

            };
            int _tmain(int argc, _TCHAR* argv[])
            {
            Base* p=new Derived();
            p->f();
            delete p;
            int i;
            cin>>i;


            return 0;
            }
              回復(fù)  更多評(píng)論   


            My Links

            Blog Stats

            常用鏈接

            留言簿(1)

            隨筆分類

            隨筆檔案

            搜索

            最新評(píng)論

            閱讀排行榜

            評(píng)論排行榜

            色偷偷88欧美精品久久久| 亚洲精品高清国产一线久久| 亚洲精品国产美女久久久| 久久久久99精品成人片试看| 久久久久人妻精品一区二区三区| 国产精品9999久久久久| 青青青国产精品国产精品久久久久| 久久香蕉国产线看观看99 | 欧洲国产伦久久久久久久| 伊色综合久久之综合久久| 久久精品www人人爽人人| 伊人久久大香线蕉影院95| 四虎国产精品成人免费久久| 热re99久久6国产精品免费| 久久97久久97精品免视看| 亚洲va久久久噜噜噜久久天堂| 91精品国产91久久久久久蜜臀 | 国内精品久久久久影院亚洲| 国产麻豆精品久久一二三| 久久亚洲中文字幕精品一区| 国产午夜久久影院| 久久一日本道色综合久久| 久久精品无码一区二区app| 久久久久亚洲精品天堂| 久久国产欧美日韩精品免费| 99久久无码一区人妻| 久久99精品久久只有精品| 亚洲综合伊人久久大杳蕉| 久久久久综合中文字幕| 国产精品久久久久久福利69堂| 国产69精品久久久久观看软件| 精品久久综合1区2区3区激情| 久久久久久毛片免费播放| 亚洲国产精品无码久久一区二区 | 日本精品久久久久影院日本 | 久久精品国产精品亚洲精品 | 亚洲va久久久噜噜噜久久男同| 欧美久久一区二区三区| 欧美无乱码久久久免费午夜一区二区三区中文字幕 | 国产精品久久久久久久久免费| 伊人久久大香线蕉亚洲|