PHP中的 抽象類(abstract class)和 接口(interface)

一、?抽象類abstract class

1?.抽象類是指在?class?前加了?abstract?關鍵字且存在抽象方法(在類方法?function?關鍵字前加了?abstract?關鍵字)的類。

2?.抽象類不能被直接實例化。抽象類中只定義(或部分實現)子類需要的方法。子類可以通過繼承抽象類并通過實現抽象類中的所有抽象方法,使抽象類具體化。

3?.如果子類需要實例化,前提是它實現了抽象類中的所有抽象方法。如果子類沒有全部實現抽象類中的所有抽象方法,那么該子類也是一個抽象類,必須在?class?前面加上?abstract?關鍵字,并且不能被實例化。

[c-sharp]?view plain?copy

abstract?class?A??

{??

/**?抽象類中可以定義變量?*/??

protected?$value1?=?0;??

private?$value2?=?1;??

public?$value3?=?2;??

/**?也可以定義非抽象方法?*/??

public?function?my_print()??

????{??

echo"hello,world/n";??

????}??

/**

?????*?大多數情況下,抽象類至少含有一個抽象方法。抽象方法用abstract關鍵字聲明,其中不能有具體內容。

?????*?可以像聲明普通類方法那樣聲明抽象方法,但是要以分號而不是方法體結束。也就是說抽象方法在抽象類中不能被實現,也就是沒有函數體“{some?codes}”。

?????*/??

abstract?protected?function?abstract_func1();??

abstract?protected?function?abstract_func2();??

}??

abstract?class?B?extends?A??

{??

public?function?abstract_func1()??

????{??

echo"implement?the?abstract_func1?in?class?A/n";??

????}??

/**?這么寫在zend?studio?8中會報錯*/??

//abstract?protected?function?abstract_func2();??

}??

class?C?extends?B??

{??

public?function?abstract_func2()??

????{??

echo"implement?the?abstract_func2?in?class?A/n";??

????}??

}??

4?.如果像下面這樣創建了一個繼承自?A?的子類?B?,但是不實現抽象方法?abstract_func()?:

[php]?view plain?copy

Class?B?extends?A{};??

那么程序將出現以下錯誤:

[php]?view plain?copy

Fatal?error:?Class?B?contains?1?abstract?method?and?must?therefore?be?declared?abstract?or?implement?the?remaining?methods?(A::abstract_func)??

5?.如果?B?實現了抽象方法?abstract_func()?,那么?B?中?abstract_func()?方法的訪問控制不能比?A?中?abstract_func()?的訪問控制更嚴格,也就是說:

(1)?如果?A?中?abstract_func()?聲明為?public?,那么?B?中?abstract_func()?的聲明只能是?public?,不能是protected?或?private

(2)?如果?A?中?abstract_func()?聲明為?protected?,那么?B?中?abstract_func()?的聲明可以是?public?或?protected,但不能是?private

(3)?如果?A?中?abstract_func()?聲明為?private?,嘿嘿,不能定義為?private?哦!(?Fatal error?: Abstract function A::abstract_func() cannot be declared private?)

二、?接口interface

1?.抽象類提供了具體實現的標準,而接口則是純粹的模版。接口只定義功能,而不包含實現的內容。接口用關鍵字interface?來聲明。

2?.?interface?是完全抽象的,只能聲明方法,而且只能聲明?public?的方法,不能聲明?private?及?protected?的方法,不能定義方法體,也不能聲明實例變量?。然而,?interface?卻可以聲明常量變量?。但將常量變量放在?interface?中違背了其作為接口的作用而存在的宗旨,也混淆了?interface?與類的不同價值。如果的確需要,可以將其放在相應的?abstract class或?Class?中。

[php]?view plain?copy

interface?iA??

{??

const?AVAR=3;??

public?function?iAfunc1();??

public?function?iAfunc2();??

}??

echo?iA::?AVAR;??


3?.任何實現接口的類都要實現接口中所定義的所有方法

[php]?view plain?copy

class?E?implements?iA??

{??

public?function?iAfunc1(){echo?"in?iAfunc1";}??

public?function?iAfunc2(){echo?"in?iAfunc2";}??

}??

否則該類必須聲明為?abstract?。

[php]?view plain?copy

abstract?class?E?implements?iA{}??

4?.一個類可以在聲明中使用?implements?關鍵字來實現某個接口。這么做之后,實現接口的具體過程和繼承一個僅包含抽象方法的抽象類是一樣的。一個類可以同時繼承一個父類和實現任意多個接口。?extends?子句應該在?implements?子句之前。?PHP?只支持繼承自一個父類,因此?extends?關鍵字后只能跟一個類名。

[php]?view plain?copy

interface?iB??

{??

public?function?iBfunc1();??

public?function?iBfunc2();??

}??

class?D?extends?A?implements?iA,iB??

{??

public?function?abstract_func1()??

????{??

echo?"implement?the?abstract_func1?in?class?A/n";??

????}??

public?function?abstract_func2()??

????{??

echo?"implement?the?abstract_func2?in?class?A/n";??

????}??

public?function?iAfunc1(){echo?"in?iAfunc1";}??

public?function?iAfunc2(){echo?"in?iAfunc2";}??

public?function?iBfunc1(){echo?"in?iBfunc1";}??

public?function?iBfunc2(){echo?"in?iBfunc2";}??

}??


class?D?extends?B?implements?iA,iB??

{??

public?function?abstract_func1()??

????{??

???????parent::abstract_func1();??

echo?"override?the?abstract_func1?in?class?A/n";??

????}??

public?function?abstract_func2()??

????{??

echo?"implement?the?abstract_func2?in?class?A/n";??

????}??

public?function?iAfunc1(){echo?"in?iAfunc1";}??

public?function?iAfunc2(){echo?"in?iAfunc2";}??

public?function?iBfunc1(){echo?"in?iBfunc1";}??

public?function?iBfunc2(){echo?"in?iBfunc2";}??

}??

5?.接口不可以實現另一個接口,但可以繼承多個

[php]?view plain?copy

interface?iC?extends?iA,iB{}??

class?F?implements?iC??

{??

public?function?iAfunc1(){echo?"in?iAfunc1";}??

public?function?iAfunc2(){echo?"in?iAfunc2";}??

public?function?iBfunc1(){echo?"in?iBfunc1";}??

public?function?iBfunc2(){echo?"in?iBfunc2";}??

}??

三、?抽象類和接口的異同

1.?相同點:

(1)?兩者都是抽象類,都不能實例化。

(2)?interface?實現類及?abstract class?的子類都必須要實現已經聲明的抽象方法。

2.?不同點:

(1)?interface?需要實現,要用?implements?,而?abstract class?需要繼承,要用?extends?。

(2)?一個類可以實現多個?interface?,但一個類只能繼承一個?abstract class?。

(3)?interface?強調特定功能的實現,而?abstract class?強調所屬關系。

(4)?盡管?interface?實現類及?abstract class?的子類都必須要實現相應的抽象方法,但實現的形式不同。?interface?中的每一個方法都是抽象方法,都只是聲明的?(declaration,?沒有方法體?)?,實現類必須要實現。而?abstract class的子類可以有選擇地實現。這個選擇有兩點含義:?a) abstract class?中并非所有的方法都是抽象的,只有那些冠有abstract?的方法才是抽象的,子類必須實現。那些沒有?abstract?的方法,在?abstract class?中必須定義方法體;b) abstract class?的子類在繼承它時,對非抽象方法既可以直接繼承,也可以覆蓋;而對抽象方法,可以選擇實現,也可以留給其子類來實現,但此類必須也聲明為抽象類。既是抽象類,當然也不能實例化。

(5)?abstract class?是?interface?與?class?的中介。?abstract class?在?interface?及?class?中起到了承上啟下的作用。一方面,?abstract class?是抽象的,可以聲明抽象方法,以規范子類必須實現的功能;另一方面,它又可以定義缺省的方法體,供子類直接使用或覆蓋。另外,它還可以定義自己的實例變量,以供子類通過繼承來使用。

(6)?接口中的抽象方法前不用也不能加?abstract?關鍵字,默認隱式就是抽象方法,也不能加?final?關鍵字來防止抽象方法的繼承。而抽象類中抽象方法前則必須加上?abstract?表示顯示聲明為抽象方法。

(7)?接口中的抽象方法默認是?public?的,也只能是?public?的,不能用?private?,?protected?修飾符修飾。而抽象類中的抽象方法則可以用?public?,?protected?來修飾,但不能用?private?。

3. interface?的應用場合

(1)?類與類之間需要特定的接口進行協調,而不在乎其如何實現。

(2)?作為能夠實現特定功能的標識存在,也可以是什么接口方法都沒有的純粹標識。

(3)?需要將一組類視為單一的類,而調用者只通過接口來與這組類發生聯系。

(4)?需要實現特定的多項功能,而這些功能之間可能完全沒有任何聯系。

4. abstract class?的應用場合

一句話,在既需要統一的接口,又需要實例變量或缺省的方法的情況下,就可以使用它。最常見的有:

(1)?定義了一組接口,但又不想強迫每個實現類都必須實現所有的接口。可以用?abstract class?定義一組方法體,甚至可以是空方法體,然后由子類選擇自己所感興趣的方法來覆蓋。

(2)?某些場合下,只靠純粹的接口不能滿足類與類之間的協調,還必需類中表示狀態的變量來區別不同的關系。abstract?的中介作用可以很好地滿足這一點。

(3)?規范了一組相互協調的方法,其中一些方法是共同的,與狀態無關的,可以共享的,無需子類分別實現;而另一些方法卻需要各個子類根據自己特定的狀態來實現特?定的功能?。

?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,546評論 6 533
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,570評論 3 418
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,505評論 0 376
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,017評論 1 313
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,786評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,219評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,287評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,438評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,971評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,796評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,995評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,540評論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,230評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,662評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,918評論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,697評論 3 392
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,991評論 2 374

推薦閱讀更多精彩內容