C#中抽象类和接口的区别与使用
一、抽象類:
抽象類是特殊的類,只是不能被實(shí)例化;除此以外,具有類的其他特性;重要的是抽象類可以包括抽象方法,這是普通類所不能的。抽象方法只能聲明于抽象類中,且不包含任何實(shí)現(xiàn),派生類必須覆蓋它們。另外,抽象類可以派生自一個(gè)抽象類,可以覆蓋基類的抽象方法也可以不覆蓋,如果不覆蓋,則其派生類必須覆蓋它們。
二、接口:
接口是引用類型的,類似于類,和抽象類的相似之處有三點(diǎn):
1、不能實(shí)例化;
2、包含未實(shí)現(xiàn)的方法聲明;
3、派生類必須實(shí)現(xiàn)未實(shí)現(xiàn)的方法,抽象類是抽象方法,接口則是所有成員(不僅是方法包括其他成員);
另外,接口有如下特性:
接口除了可以包含方法之外,還可以包含屬性、索引器、事件,而且這些成員都被定義為公有的。除此之外,不能包含任何其他的成員,例如:常量、域、構(gòu)造函數(shù)、析構(gòu)函數(shù)、靜態(tài)成員。一個(gè)類可以直接繼承多個(gè)接口,但只能直接繼承一個(gè)類(包括抽象類)。
三、抽象類和接口的區(qū)別:
1.類是對(duì)對(duì)象的抽象,可以把抽象類理解為把類當(dāng)作對(duì)象,抽象成的類叫做抽象類.而接口只是一個(gè)行為的規(guī)范或規(guī)定,微軟的自定義接口總是后帶able字段,證明其是表述一類類“我能做。。。”.抽象類更多的是定義在一系列緊密相關(guān)的類間,而接口大多數(shù)是關(guān)系疏松但都實(shí)現(xiàn)某一功能的類中.
2.接口基本上不具備繼承的任何具體特點(diǎn),它僅僅承諾了能夠調(diào)用的方法;????
3.一個(gè)類一次可以實(shí)現(xiàn)若干個(gè)接口,但是只能擴(kuò)展一個(gè)父類????
4.接口可以用于支持回調(diào),而繼承并不具備這個(gè)特點(diǎn).????
5.抽象類不能被密封。??
6.抽象類實(shí)現(xiàn)的具體方法默認(rèn)為虛的,但實(shí)現(xiàn)接口的類中的接口方法卻默認(rèn)為非虛的,當(dāng)然您也可以聲明為虛的.
7.(接口)與非抽象類類似,抽象類也必須為在該類的基類列表中列出的接口的所有成員提供它自己的實(shí)現(xiàn)。但是,允許抽象類將接口方法映射到抽象方法上。??
8.抽象類實(shí)現(xiàn)了oop中的一個(gè)原則,把可變的與不可變的分離。抽象類和接口就是定義為不可變的,而把可變的座位子類去實(shí)現(xiàn)。
9.好的接口定義應(yīng)該是具有專一功能性的,而不是多功能的,否則造成接口污染。如果一個(gè)類只是實(shí)現(xiàn)了這個(gè)接口的中一個(gè)功能,而不得不去實(shí)現(xiàn)接口中的其他方法,就叫接口污染。??
10.盡量避免使用繼承來(lái)實(shí)現(xiàn)組建功能,而是使用黑箱復(fù)用,即對(duì)象組合。因?yàn)槔^承的層次增多,造成最直接的后果就是當(dāng)你調(diào)用這個(gè)類群中某一類,就必須把他們?nèi)考虞d到棧中!后果可想而知.(結(jié)合堆棧原理理解)。同時(shí),有心的朋友可以留意到微軟在構(gòu)建一個(gè)類時(shí),很多時(shí)候用到了對(duì)象組合的方法。比如asp.net中,Page類,有Server Request等屬性,但其實(shí)他們都是某個(gè)類的對(duì)象。使用Page類的這個(gè)對(duì)象來(lái)調(diào)用另外的類的方法和屬性,這個(gè)是非常基本的一個(gè)設(shè)計(jì)原則。??
11.如果抽象類實(shí)現(xiàn)接口,則可以把接口中方法映射到抽象類中作為抽象方法而不必實(shí)現(xiàn),而在抽象類的子類中實(shí)現(xiàn)接口中方法.
四、抽象類和接口的使用:
1. 如果預(yù)計(jì)要?jiǎng)?chuàng)建組件的多個(gè)版本,則創(chuàng)建抽象類。抽象類提供簡(jiǎn)單的方法來(lái)控制組件版本。
2.如果創(chuàng)建的功能將在大范圍的全異對(duì)象間使用,則使用接口。如果要設(shè)計(jì)小而簡(jiǎn)練的功能塊,則使用接口。
3.如果要設(shè)計(jì)大的功能單元,則使用抽象類.如果要在組件的所有實(shí)現(xiàn)間提供通用的已實(shí)現(xiàn)功能,則使用抽象類。??
4.抽象類主要用于關(guān)系密切的對(duì)象;而接口適合為不相關(guān)的類提供通用功能。
以下是我在網(wǎng)上看到的幾個(gè)形象比喻,真的非常不錯(cuò),呵呵:
1.飛機(jī)會(huì)飛,鳥會(huì)飛,他們都繼承了同一個(gè)接口“飛”;但是F22屬于飛機(jī)抽象類,鴿子屬于鳥抽象類。
2. 就像鐵門木門都是門(抽象類),你想要個(gè)門我給不了(不能實(shí)例化),但我可以給你個(gè)具體的鐵門或木門(多態(tài));而且只能是門,你不能說(shuō)它是窗(單繼承);一個(gè)門可以有鎖(接口)也可以有門鈴(多實(shí)現(xiàn))。門(抽象類)定義了你是什么,接口(鎖)規(guī)定了你能做什么(一個(gè)接口最好只能做一件事,你不能要求鎖也能發(fā)出聲音吧(接口污染)。
?
?
http://kb.cnblogs.com/page/41836/
?
Net提供了接口,這個(gè)不同于Class或者Struct的類型定義。接口有些情況,看似和抽象類一樣,因此有些人認(rèn)為在.Net可以完全用接口來(lái)替換抽象類。其實(shí)不然,接口和抽象類各有長(zhǎng)處和缺陷,因此往往在應(yīng)用當(dāng)中,兩者要結(jié)合來(lái)使用,從而互補(bǔ)長(zhǎng)短。
接下來(lái)先說(shuō)說(shuō)抽象類和接口的區(qū)別。
區(qū)別一,兩者表達(dá)的概念不一樣。抽象類是一類事物的高度聚合,那么對(duì)于繼承抽象類的子類來(lái)說(shuō),對(duì)于抽象類來(lái)說(shuō),屬于“是”的關(guān)系;而接口是定義行為規(guī)范,因此對(duì)于實(shí)現(xiàn)接口的子類來(lái)說(shuō),相對(duì)于接口來(lái)說(shuō),是“行為需要按照接口來(lái)完成”。這些聽(tīng)起來(lái)有些虛,舉個(gè)例子。例如,狗是對(duì)于所有狗類動(dòng)物的統(tǒng)稱,京哈是狗,牧羊犬是狗,那么狗的一般特性,都會(huì)在京哈,牧羊犬中找到,那么狗相對(duì)于京哈和牧羊犬來(lái)說(shuō),就屬于這類事物的抽象類型;而對(duì)于“叫”這個(gè)動(dòng)作來(lái)說(shuō),狗可以叫,鳥也可以叫。很明顯,前者相當(dāng)于所說(shuō)的是抽象類,而后者指的就是接口。
區(qū)別二,抽象類在定義類型方法的時(shí)候,可以給出方法的實(shí)現(xiàn)部分,也可以不給出;而對(duì)于接口來(lái)說(shuō),其中所定義的方法都不能給出實(shí)現(xiàn)部分。
例如:
????public abstract class AbsTest
????{
????????public virtual void Test()
????????{
????????????Debug.WriteLine( "Test" );
????????}
????????public abstract void NewTest();
????}
????public interface ITest
????{
????????void Test();
????????void NewTest();
????}
區(qū)別三,繼承類對(duì)于兩者所涉及方法的實(shí)現(xiàn)是不同的。繼承類對(duì)于抽象類所定義的抽象方法,可以不用重寫,也就是說(shuō),可以延用抽象類的方法;而對(duì)于接口類所定義的方法或者屬性來(lái)說(shuō),在繼承類中必須要給出相應(yīng)的方法和屬性實(shí)現(xiàn)。
區(qū)別四,在抽象類中,新增一個(gè)方法的話,繼承類中可以不用作任何處理;而對(duì)于接口來(lái)說(shuō),則需要修改繼承類,提供新定義的方法。
知道了兩者的區(qū)別,再來(lái)說(shuō)說(shuō),接口相對(duì)于抽象類的優(yōu)勢(shì)。
好處一,接口不光可以作用于引用類型,也可以作用于值類型。而抽象類來(lái)說(shuō),只能作用于引用類型。
好處二,.Net的類型繼承只能是單繼承的,也就是說(shuō)一個(gè)類型只能繼承一個(gè)類型,而可以繼承多個(gè)接口。其實(shí),我對(duì)于這一點(diǎn)也比較贊同,多繼承會(huì)使繼承樹變的混亂。
好處三,由于接口只是定義屬性和方法,而與真正實(shí)現(xiàn)的類型沒(méi)有太大的關(guān)系,因此接口可以被多個(gè)類型重用。相對(duì)于此,抽象類與繼承類的關(guān)系更緊密些。
好處四,通過(guò)接口,可以減少類型暴露的屬性和方法,從而便于保護(hù)類型對(duì)象。當(dāng)一個(gè)實(shí)現(xiàn)接口的類型,可能包含其他方法或者屬性,但是方法返回的時(shí)候,可以返回接口對(duì)象,這樣調(diào)用端,只能通過(guò)接口提供的方法或者屬性,訪問(wèn)對(duì)象的相關(guān)元素,這樣可以有效保護(hù)對(duì)象的其他元素。
好處五,減少值類型的拆箱操作。對(duì)于Struct定義的值類型數(shù)據(jù),當(dāng)存放集合當(dāng)中,每當(dāng)取出來(lái),都需要進(jìn)行拆箱操作,這時(shí)采用Struct+Interface結(jié)合的方法,從而降低拆箱操作。
參看如下文章提供的方法。
http://blog.csdn.net/Knight94/archive/2006/10/08/1326326.aspx
相對(duì)于抽象類來(lái)說(shuō),接口有這么多好處,但是接口有一個(gè)致命的弱點(diǎn),就是接口所定義的方法和屬性只能相對(duì)于繼承它的類型(除非在繼承類中修改借口定義的函數(shù)標(biāo)示),那么對(duì)于多層繼承關(guān)系的時(shí)候,光用接口就很難實(shí)現(xiàn)。因?yàn)槿绻屆總€(gè)類型都去繼承接口而進(jìn)行實(shí)現(xiàn)的話,首先不說(shuō)編寫代碼比較繁瑣,有時(shí)候執(zhí)行的結(jié)果還是錯(cuò)誤,尤其當(dāng)子類型對(duì)象隱式轉(zhuǎn)換成基類對(duì)象進(jìn)行訪問(wèn)的時(shí)候。
那么這時(shí)候,需要用接口結(jié)合虛方法來(lái)實(shí)現(xiàn)。參看IDisposable在繼承類型中的實(shí)現(xiàn)方法。
http://blog.csdn.net/Knight94/archive/2006/10/10/1329214.aspx
其實(shí)在繼承中,到底使用接口還是抽象類。接口是固定的,約定俗成的,因此在繼承類中必須提供接口相應(yīng)的方法和屬性的實(shí)現(xiàn)。而對(duì)于抽象類來(lái)說(shuō),抽象類的定義方法的實(shí)現(xiàn),貫穿整個(gè)繼承樹,因此其中方法的實(shí)現(xiàn)或者重寫都是不確定的。因此相對(duì)而言,抽象類比接口更靈活一些。
如下給出兩者的簡(jiǎn)單對(duì)比表格。
??接口 抽象類
多繼承 支持 不支持
類型限制 沒(méi)有 有,只能是引用類型
方法實(shí)現(xiàn) 繼承類型中必須給出方法實(shí)現(xiàn) 繼承類中可以不給出
擴(kuò)展性 比較麻煩 相對(duì)比較靈活
多層繼承 比較麻煩,需要借助虛函數(shù) 比較靈活
總的來(lái)說(shuō),接口和抽象類是.Net為了更好的實(shí)現(xiàn)類型之間繼承關(guān)系而提供的語(yǔ)言手段,而且兩者有些相輔相成的關(guān)系。因此我并不強(qiáng)調(diào)用什么而不用什么,那么問(wèn)題的關(guān)鍵在于,如何把這兩種手段合理的應(yīng)用到程序當(dāng)中,這才是至關(guān)重要。
轉(zhuǎn)載于:https://www.cnblogs.com/hedianzhan/p/8854263.html
總結(jié)
以上是生活随笔為你收集整理的C#中抽象类和接口的区别与使用的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: glusterfs4.0.1 mempo
- 下一篇: 看jquery3.3.1学js类型判断的