抽象類 & 接口
一、抽象類:
抽象類是特殊的類,只是不能被實例化;除此以外,具有類的其他特性;重要的是抽象類可以包括抽象方法,這是普通類所不能的。抽象方法只能聲明于抽象類中,且不包含任何實現(xiàn),派生類必須覆蓋它們。另外,抽象類可以派生自一個抽象類,可以覆蓋基類的抽象方法也可以不覆蓋,如果不覆蓋,則其派生類必須覆蓋它們。
二、接口:
接口是引用類型的,類似于類,和抽象類的相似之處有三點:
1、不能實例化;
2、包含未實現(xiàn)的方法聲明;
3、派生類必須實現(xiàn)未實現(xiàn)的方法,抽象類是抽象方法,接口則是所有成員(不僅是方法包括其他成員);
另外,接口有如下特性:
接口除了可以包含方法之外,還可以包含屬性、索引器、事件,而且這些成員都被定義為公有的。除此之外,不能包含任何其他的成員,例如:常量、域、構(gòu)造函數(shù)、析構(gòu)函數(shù)、靜態(tài)成員。一個類可以直接繼承多個接口,但只能直接繼承一個類(包括抽象類)。
三、抽象類和接口的區(qū)別:
1.類是對對象的抽象,可以把抽象類理解為把類當(dāng)作對象,抽象成的類叫做抽象類.而接口只是一個行為的規(guī)范或規(guī)定,微軟的自定義接口總是后帶able字段,證明其是表述一類類“我能做。。。”.抽象類更多的是定義在一系列緊密相關(guān)的類間,而接口大多數(shù)是關(guān)系疏松但都實現(xiàn)某一功能的類中.
2.接口基本上不具備繼承的任何具體特點,它僅僅承諾了能夠調(diào)用的方法;
3.一個類一次可以實現(xiàn)若干個接口,但是只能擴(kuò)展一個父類
4.接口可以用于支持回調(diào),而繼承并不具備這個特點.
5.抽象類不能被密封。
6.抽象類實現(xiàn)的具體方法默認(rèn)為虛的,但實現(xiàn)接口的類中的接口方法卻默認(rèn)為非虛的,當(dāng)然您也可以聲明為虛的.
7.(接口)與非抽象類類似,抽象類也必須為在該類的基類列表中列出的接口的所有成員提供它自己的實現(xiàn)。但是,允許抽象類將接口方法映射到抽象方法上。
8.抽象類實現(xiàn)了oop中的一個原則,把可變的與不可變的分離。抽象類和接口就是定義為不可變的,而把可變的座位子類去實現(xiàn)。
9.好的接口定義應(yīng)該是具有專一功能性的,而不是多功能的,否則造成接口污染。如果一個類只是實現(xiàn)了這個接口的中一個功能,而不得不去實現(xiàn)接口中的其他方法,就叫接口污染。
10.盡量避免使用繼承來實現(xiàn)組建功能,而是使用黑箱復(fù)用,即對象組合。因為繼承的層次增多,造成最直接的后果就是當(dāng)你調(diào)用這個類群中某一類,就必須把他們?nèi)考虞d到棧中!后果可想而知.(結(jié)合堆棧原理理解)。同時,有心的朋友可以留意到微軟在構(gòu)建一個類時,很多時候用到了對象組合的方法。比如asp.net中,Page類,有Server Request等屬性,但其實他們都是某個類的對象。使用Page類的這個對象來調(diào)用另外的類的方法和屬性,這個是非常基本的一個設(shè)計原則。
11.如果抽象類實現(xiàn)接口,則可以把接口中方法映射到抽象類中作為抽象方法而不必實現(xiàn),而在抽象類的子類中實現(xiàn)接口中方法.
虛函數(shù)、純虛函數(shù)、非虛函數(shù)
虛函數(shù)和純虛函數(shù)有以下所示方面的區(qū)別。
(1)類里如果聲明了虛函數(shù),這個函數(shù)是實現(xiàn)的,哪怕是空實現(xiàn),它的作用就是為了能讓這個函數(shù)在它的子類里面可以被覆蓋,這樣的話,這樣編譯器就可以使用后期綁定來達(dá)到多態(tài)了。純虛函數(shù)只是一個接口,是個函數(shù)的聲明而已,它要留到子類里去實現(xiàn)。
(2)虛函數(shù)在子類里面也可以不重載的;但純虛函數(shù)必須在子類去實現(xiàn),這就像Java的接口一樣。通常把很多函數(shù)加上virtual,是一個好的習(xí)慣,雖然犧牲了一些性能,但是增加了面向?qū)ο蟮亩鄳B(tài)性,因為很難預(yù)料到父類里面的這個函數(shù)不在子類里面不去修改它的實現(xiàn)。
(3)虛函數(shù)的類用于“實作繼承”,繼承接口的同時也繼承了父類的實現(xiàn)。當(dāng)然大家也可以完成自己的實現(xiàn)。純虛函數(shù)關(guān)注的是接口的統(tǒng)一性,實現(xiàn)由子類完成。
(4)帶純虛函數(shù)的類叫虛基類,這種基類不能直接生成對象,而只有被繼承,并重寫其虛函數(shù)后,才能使用。這樣的類也叫抽象類。抽象類和大家口頭常說的虛基類還是有區(qū)別的,在C#中用abstract定義抽象類,而在C++中有抽象類的概念,但是沒有這個關(guān)鍵字。抽象類被繼承后,子類可以繼續(xù)是抽象類,也可以是普通類,而虛基類,是含有純虛函數(shù)的類,它如果被繼承,那么子類就必須實現(xiàn)虛基類里面的所有純虛函數(shù),其子類不能是抽象類。
純虛函數(shù)
聲明了純虛函數(shù)的類是一個抽象類。所以,用戶不能創(chuàng)建類的實例,只能創(chuàng)建它的派生類的實例。
純虛函數(shù)最顯著的特征是:它們必須在繼承類中重新聲明函數(shù)(不要后面的=0,否則該派生類也不能實例化),而且它們在抽象類中往往沒有定義。
定義純虛函數(shù)的目的在于,使派生類僅僅只是繼承函數(shù)的接口。
純虛函數(shù)的意義,讓所有的類對象(主要是派生類對象)都可以執(zhí)行純虛函數(shù)的動作,但類無法為純虛函數(shù)提供一個合理的缺省實現(xiàn)。所以類純虛函數(shù)的聲明就是在告訴子類的設(shè)計者,“你必須提供一個純虛函數(shù)的實現(xiàn),但我不知道你會怎樣實現(xiàn)它”。
順便說一句,為一個純虛函數(shù)提供定義也是可能的。也就是說,你可以為純虛函數(shù)提供實現(xiàn),C++編譯器也不會阻攔(DEV_CPP中G++(gcc 3.4.2)編譯器并不支持為純虛函數(shù)定義缺省行為;在VC6.0支持為純虛函數(shù)定義缺省的實現(xiàn),派生類的虛函數(shù)override基類的純虛函數(shù)),但調(diào)用它的唯一方式是通過類名完整地指明是哪個調(diào)用(如:pb->Base:: pureVirtual())。
有時,聲明一個除純虛函數(shù)外什么也不包含的類很有用。這樣的類叫協(xié)議類(Protocol class),它為派生類僅提供函數(shù)接口,完全沒有實現(xiàn)。
虛函數(shù)(在此指的是非純虛函數(shù))
虛函數(shù)的情況和純虛函數(shù)有點不一樣。照例,派生類繼承了函數(shù)的接口,但簡單虛函數(shù)一般還提供了實現(xiàn),派生類可以選擇改寫(override)它們或不改寫它們。
聲明虛函數(shù)的目的在于,使派生類繼承函數(shù)的接口和缺省實現(xiàn)。
虛函數(shù)的意義,每個類必須提供一個可以被調(diào)用的虛函數(shù),但每個類可以按它們認(rèn)為合適的任何方式處理。如果某個類不想做什么特別的事,可以借助于基類中提供的缺省處理函數(shù)。也就是說,虛函數(shù)的聲明是在告訴子類的設(shè)計者,"你必須支持虛函數(shù),但如果你不想寫自己的版本,可以借助基類中的缺省版本。"
實際上,為虛函數(shù)同時提供函數(shù)聲明和缺省實現(xiàn)是很危險的。(當(dāng)你增加一個派生類繼承基類時,必須小心使用虛函數(shù),滿足派生類特有的需求,否則就是調(diào)用基類的虛函數(shù),可能引起錯誤)
非虛函數(shù)
最后,來談?wù)勵惖姆翘摵瘮?shù),當(dāng)一個成員函數(shù)為非虛函數(shù)時,它在派生類中的行為就不應(yīng)該不同。實際上,非虛成員函數(shù)表明了一種特殊性上的不變性,因為它表示的是不會改變的行為――不管一個派生類有多特殊。
聲明非虛函數(shù)的目的在于,使派生類繼承函數(shù)的接口和強(qiáng)制性實現(xiàn)。(所有的派生類都應(yīng)該完成的使用該函數(shù)完成某一個功能)
建議
結(jié)合前面的學(xué)過的,再次強(qiáng)調(diào)一下,如果你沒有為類設(shè)計虛函數(shù)(純虛函數(shù)),該類一般來說應(yīng)該不具有繼承特性(除非確實的存在IS-A關(guān)系,即便存在,派生類也沒有了特殊性,這種情況一般是設(shè)計中抽象的不合理)。當(dāng)然除了Protocol class也不應(yīng)該把類的成員函數(shù)全部設(shè)計成虛函數(shù)(純虛函數(shù)),這也說明了類設(shè)計的不合理(不能正確的抽象出基類、派生類之間不變的部分)。
本文來自CSDN博客,轉(zhuǎn)載請標(biāo)明出處:http://blog.csdn.net/bmzyDream_007/archive/2009/05/07/4157560.aspx