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

            技術無極限

            常用鏈接

            統計

            最新評論

            多態~~我一直沒有形成理論化的知識

            多態性,這個面向對象編程領域的核心概念,本身的內容博大精深,要以一文說清楚實在是不太可能。加之作者本人也還在不斷學習中,水平有限。因此本文只能描一下多態的輪廓,使讀者能夠了解個大概。如果有描的不準的地方,歡迎指出,或與作者探討(作者Email:nicrosoft@sunistudio.com)  
                       
                      首先,什么是多態(Polymorphisn)?按字面的意思就是“多種形狀”。我手頭的書上沒有找到一個多態的理論性的概念的描述。暫且引用一下Charlie   Calverts的對多態的描述吧——多態性是允許你將父對象設置成為和一個或更多的他的子對象相等的技術,賦值之后,父對象就可以根據當前賦值給它的子對象的特性以不同的方式運作(摘自“Delphi4   編程技術內幕”)。簡單的說,就是一句話:允許將子類類型的指針賦值給父類類型的指針。多態性在Object   Pascal和C++中都是通過虛函數(Virtual   Function)實現的。  
                       
                      好,接著是“虛函數”(或者是“虛方法”)。虛函數就是允許被其子類重新定義的成員函數。而子類重新定義父類虛函數的做法,稱為“覆蓋”(override),或者稱為“重寫”。  
               
                      這里有一個初學者經常混淆的概念。覆蓋(override)和重載(overload)。上面說了,覆蓋是指子類重新定義父類的虛函數的做法。而重載,是指允許存在多個同名函數,而這些函數的參數表不同(或許參數個數不同,或許參數類型不同,或許兩者都不同)。其實,重載的概念并不屬于“面向對象編程”,重載的實現是:編譯器根據函數不同的參數表,對同名函數的名稱做修飾,然后這些同名函數就成了不同的函數(至少對于編譯器來說是這樣的)。如,有兩個同名函數:function   func(p:integer):integer;和function   func(p:string):integer;。那么編譯器做過修飾后的函數名稱可能是這樣的:int_func、str_func。對于這兩個函數的調用,在編譯器間就已經確定了,是靜態的(記住:是靜態)。也就是說,它們的地址在編譯期就綁定了(早綁定),因此,重載和多態無關!真正和多態相關的是“覆蓋”。當子類重新定義了父類的虛函數后,父類指針根據賦給它的不同的子類指針,動態(記住:是動態!)的調用屬于子類的該函數,這樣的函數調用在編譯期間是無法確定的(調用的子類的虛函數的地址無法給出)。因此,這樣的函數地址是在運行期綁定的(晚邦定)。結論就是:重載只是一種語言特性,與多態無關,與面向對象也無關!  
                       
                      引用一句Bruce   Eckel的話:“不要犯傻,如果它不是晚邦定,它就不是多態。”  
                       
                      那么,多態的作用是什么呢?我們知道,封裝可以隱藏實現細節,使得代碼模塊化;繼承可以擴展已存在的代碼模塊(類);它們的目的都是為了——代碼重用。而多態則是為了實現另一個目的——接口重用!而且現實往往是,要有效重用代碼很難,而真正最具有價值的重用是接口重用,因為“接口是公司最有價值的資源。設計接口比用一堆類來實現這個接口更費時間。而且接口需要耗費更昂貴的人力的時間。”  
                       
                      其實,繼承的為重用代碼而存在的理由已經越來越薄弱,因為“組合”可以很好的取代繼承的擴展現有代碼的功能,而且“組合”的表現更好(至少可以防止“類爆炸”)。因此筆者個人認為,繼承的存在很大程度上是作為“多態”的基礎而非擴展現有代碼的方式了。  
                       
                      什么是接口重用?我們舉一個簡單的例子,假設我們有一個描述飛機的基類(Object   Pascal語言描述,下同):  
                      type  
                              plane   =   class  
                              public  
                                      procedure   fly();   virtual;   abstract;   //起飛純虛函數  
                                      procedure   land();   virtual;   abstract;   //著陸純虛函數  
                                      function   modal()   :   string;   virtual;   abstract;   //查尋型號純虛函數  
                              end;  
                       
                      然后,我們從plane派生出兩個子類,直升機(copter)和噴氣式飛機(jet):  
                              copter   =   class(plane)  
                              private  
                                      fModal   :   String;  
                              public  
                                      constructor   Create();  
                                      destructor   Destroy();   override;  
                                      procedure   fly();   override;  
                                      procedure   land();   override;  
                                      function   modal()   :   string;   override;  
                              end;  
                       
                              jet   =   class(plane)  
                              private  
                                      fModal   :   String;  
                              public  
                                      constructor   Create();  
                                      destructor   Destroy();   override;  
                                      procedure   fly();   override;  
                                      procedure   land();   override;  
                                      function   modal()   :   string;   override;  
                              end;  
                       
                      現在,我們要完成一個飛機控制系統,有一個全局的函數   plane_fly,它負責讓傳遞給它的飛機起飛,那么,只需要這樣:  
                      procedure   plane_fly(const   pplane   :   plane);  
                      begin  
                              pplane.fly();  
                      end;  
                      就可以讓所有傳給它的飛機(plane的子類對象)正常起飛!不管是直升機還是噴氣機,甚至是現在還不存在的,以后會增加的飛碟。因為,每個子類都已經定義了自己的起飛方式。  
                       
                      可以看到   plane_fly函數接受參數的是   plane類對象引用,而實際傳遞給它的都是   plane的子類對象,現在回想一下開頭所描述的“多態”:多態性是允許你將父對象設置成為和一個或更多的他的子對象相等的技術,賦值之后,父對象就可以根據當前賦值給它的子對象的特性以不同的方式運作。  
                       
                      很顯然,parent   =   child;   就是多態的實質!因為直升機“是一種”飛機,噴氣機也“是一種”飛機,因此,所有對飛機的操作,都可以對它們操作,此時,飛機類就作為一種接口。  
                       
                      多態的本質就是將子類類型的指針賦值給父類類型的指針(在OP中是引用),只要這樣的賦值發生了,多態也就產生了,因為實行了“向上映射”。  
                       
                      應用多態的例子非常普遍,在Delphi的VCL類庫中,最典型的就是:TObject類有一個虛擬的Destroy虛構函數和一個非虛擬的Free函數。Free函數中是調用Destroy的。因此,當我們對任何對象(都是TObject的子類對象)調用   .Free();之后,都會執行   TObject.Free();,它會調用我們所使用的對象的析構函數   Destroy();。這就保證了任何類型的對象都可以正確地被析構。  
               
                      多態性作為面向對象最重要的特性,本文所提不過是滄海一粟,還有很多內容。如果可能,希望會有后文繼續探討多態

            posted on 2010-01-19 09:39 勇敢空心人 閱讀(345) 評論(1)  編輯 收藏 引用 所屬分類: C/C++/算法

            評論

            # re: 多態~~我一直沒有形成理論化的知識 2010-01-31 22:10 不想成名

            C++的重要特性:多態,虛函數,重載,模板  回復  更多評論   

            久久人人爽人人爽人人片AV东京热| 久久精品免费大片国产大片 | 无码日韩人妻精品久久蜜桃| 久久永久免费人妻精品下载| 成人亚洲欧美久久久久| 香港aa三级久久三级老师2021国产三级精品三级在 | 久久强奷乱码老熟女网站| 久久91精品国产91久| 97久久超碰国产精品2021| 欧美午夜A∨大片久久 | 四虎国产精品成人免费久久| 2022年国产精品久久久久| 无码任你躁久久久久久久| 久久久九九有精品国产| 中文精品99久久国产 | 伊人久久大香线蕉无码麻豆| 久久人人爽人人爽人人片av高请 | 国产精品久久久久影院色| 久久综合鬼色88久久精品综合自在自线噜噜 | 亚洲乱码日产精品a级毛片久久| 久久久久久亚洲精品不卡| 久久久久久久久无码精品亚洲日韩| 欧美精品丝袜久久久中文字幕| 精品综合久久久久久97超人| 亚洲国产欧美国产综合久久| 久久这里只有精品首页| 午夜视频久久久久一区| 久久精品亚洲男人的天堂| 韩国免费A级毛片久久| 亚洲AV无码久久精品成人 | 亚洲狠狠婷婷综合久久久久| 色播久久人人爽人人爽人人片aV| 成人免费网站久久久| 久久天堂AV综合合色蜜桃网 | 久久婷婷五月综合国产尤物app| 狠狠色丁香婷婷综合久久来来去 | 日韩精品久久久久久久电影| 久久婷婷五月综合色99啪ak| 久久久久综合国产欧美一区二区| 久久99热这里只有精品国产| 久久国产成人午夜AV影院|