一、?抽象類abstract class
1?.抽象類是指在?class?前加了?abstract?關(guān)鍵字且存在抽象方法(在類方法?function?關(guān)鍵字前加了?abstract?關(guān)鍵字)的類。
2?.抽象類不能被直接實例化。抽象類中只定義(或部分實現(xiàn))子類需要的方法。子類可以通過繼承抽象類并通過實現(xiàn)抽象類中的所有抽象方法,使抽象類具體化。
3?.如果子類需要實例化,前提是它實現(xiàn)了抽象類中的所有抽象方法。如果子類沒有全部實現(xiàn)抽象類中的所有抽象方法,那么該子類也是一個抽象類,必須在?class?前面加上?abstract?關(guān)鍵字,并且不能被實例化。
[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";??
????}??
/**
?????*?大多數(shù)情況下,抽象類至少含有一個抽象方法。抽象方法用abstract關(guān)鍵字聲明,其中不能有具體內(nèi)容。
?????*?可以像聲明普通類方法那樣聲明抽象方法,但是要以分號而不是方法體結(jié)束。也就是說抽象方法在抽象類中不能被實現(xiàn),也就是沒有函數(shù)體“{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?.如果像下面這樣創(chuàng)建了一個繼承自?A?的子類?B?,但是不實現(xiàn)抽象方法?abstract_func()?:
[php]?view plain?copy
Class?B?extends?A{};??
那么程序?qū)⒊霈F(xiàn)以下錯誤:
[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?實現(xiàn)了抽象方法?abstract_func()?,那么?B?中?abstract_func()?方法的訪問控制不能比?A?中?abstract_func()?的訪問控制更嚴(yán)格,也就是說:
(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?.抽象類提供了具體實現(xiàn)的標(biāo)準(zhǔn),而接口則是純粹的模版。接口只定義功能,而不包含實現(xiàn)的內(nèi)容。接口用關(guān)鍵字interface?來聲明。
2?.?interface?是完全抽象的,只能聲明方法,而且只能聲明?public?的方法,不能聲明?private?及?protected?的方法,不能定義方法體,也不能聲明實例變量?。然而,?interface?卻可以聲明常量變量?。但將常量變量放在?interface?中違背了其作為接口的作用而存在的宗旨,也混淆了?interface?與類的不同價值。如果的確需要,可以將其放在相應(yīng)的?abstract class或?Class?中。
[php]?view plain?copy
interface?iA??
{??
const?AVAR=3;??
public?function?iAfunc1();??
public?function?iAfunc2();??
}??
echo?iA::?AVAR;??
3?.任何實現(xiàn)接口的類都要實現(xiàn)接口中所定義的所有方法
[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?關(guān)鍵字來實現(xiàn)某個接口。這么做之后,實現(xiàn)接口的具體過程和繼承一個僅包含抽象方法的抽象類是一樣的。一個類可以同時繼承一個父類和實現(xiàn)任意多個接口。?extends?子句應(yīng)該在?implements?子句之前。?PHP?只支持繼承自一個父類,因此?extends?關(guān)鍵字后只能跟一個類名。
[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?.接口不可以實現(xiàn)另一個接口,但可以繼承多個
[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?實現(xiàn)類及?abstract class?的子類都必須要實現(xiàn)已經(jīng)聲明的抽象方法。
2.?不同點:
(1)?interface?需要實現(xiàn),要用?implements?,而?abstract class?需要繼承,要用?extends?。
(2)?一個類可以實現(xiàn)多個?interface?,但一個類只能繼承一個?abstract class?。
(3)?interface?強(qiáng)調(diào)特定功能的實現(xiàn),而?abstract class?強(qiáng)調(diào)所屬關(guān)系。
(4)?盡管?interface?實現(xiàn)類及?abstract class?的子類都必須要實現(xiàn)相應(yīng)的抽象方法,但實現(xiàn)的形式不同。?interface?中的每一個方法都是抽象方法,都只是聲明的?(declaration,?沒有方法體?)?,實現(xiàn)類必須要實現(xiàn)。而?abstract class的子類可以有選擇地實現(xiàn)。這個選擇有兩點含義:?a) abstract class?中并非所有的方法都是抽象的,只有那些冠有abstract?的方法才是抽象的,子類必須實現(xiàn)。那些沒有?abstract?的方法,在?abstract class?中必須定義方法體;b) abstract class?的子類在繼承它時,對非抽象方法既可以直接繼承,也可以覆蓋;而對抽象方法,可以選擇實現(xiàn),也可以留給其子類來實現(xiàn),但此類必須也聲明為抽象類。既是抽象類,當(dāng)然也不能實例化。
(5)?abstract class?是?interface?與?class?的中介。?abstract class?在?interface?及?class?中起到了承上啟下的作用。一方面,?abstract class?是抽象的,可以聲明抽象方法,以規(guī)范子類必須實現(xiàn)的功能;另一方面,它又可以定義缺省的方法體,供子類直接使用或覆蓋。另外,它還可以定義自己的實例變量,以供子類通過繼承來使用。
(6)?接口中的抽象方法前不用也不能加?abstract?關(guān)鍵字,默認(rèn)隱式就是抽象方法,也不能加?final?關(guān)鍵字來防止抽象方法的繼承。而抽象類中抽象方法前則必須加上?abstract?表示顯示聲明為抽象方法。
(7)?接口中的抽象方法默認(rèn)是?public?的,也只能是?public?的,不能用?private?,?protected?修飾符修飾。而抽象類中的抽象方法則可以用?public?,?protected?來修飾,但不能用?private?。
3. interface?的應(yīng)用場合
(1)?類與類之間需要特定的接口進(jìn)行協(xié)調(diào),而不在乎其如何實現(xiàn)。
(2)?作為能夠?qū)崿F(xiàn)特定功能的標(biāo)識存在,也可以是什么接口方法都沒有的純粹標(biāo)識。
(3)?需要將一組類視為單一的類,而調(diào)用者只通過接口來與這組類發(fā)生聯(lián)系。
(4)?需要實現(xiàn)特定的多項功能,而這些功能之間可能完全沒有任何聯(lián)系。
4. abstract class?的應(yīng)用場合
一句話,在既需要統(tǒng)一的接口,又需要實例變量或缺省的方法的情況下,就可以使用它。最常見的有:
(1)?定義了一組接口,但又不想強(qiáng)迫每個實現(xiàn)類都必須實現(xiàn)所有的接口??梢杂?abstract class?定義一組方法體,甚至可以是空方法體,然后由子類選擇自己所感興趣的方法來覆蓋。
(2)?某些場合下,只靠純粹的接口不能滿足類與類之間的協(xié)調(diào),還必需類中表示狀態(tài)的變量來區(qū)別不同的關(guān)系。abstract?的中介作用可以很好地滿足這一點。
(3)?規(guī)范了一組相互協(xié)調(diào)的方法,其中一些方法是共同的,與狀態(tài)無關(guān)的,可以共享的,無需子類分別實現(xiàn);而另一些方法卻需要各個子類根據(jù)自己特定的狀態(tài)來實現(xiàn)特?定的功能?。