php接口和抽象類(lèi)使用示例詳解
一、 抽象類(lèi)abstract class
1 .抽象類(lèi)是指在 class 前加了 abstract 關(guān)鍵字且存在抽象方法(在類(lèi)方法 function 關(guān)鍵字前加了 abstract 關(guān)鍵字)的類(lèi)。
2 .抽象類(lèi)不能被直接實(shí)例化。抽象類(lèi)中只定義(或部分實(shí)現(xiàn))子類(lèi)需要的方法。子類(lèi)可以通過(guò)繼承抽象類(lèi)并通過(guò)實(shí)現(xiàn)抽象類(lèi)中的所有抽象方法,使抽象類(lèi)具體化。
3 .如果子類(lèi)需要實(shí)例化,前提是它實(shí)現(xiàn)了抽象類(lèi)中的所有抽象方法。如果子類(lèi)沒(méi)有全部實(shí)現(xiàn)抽象類(lèi)中的所有抽象方法,那么該子類(lèi)也是一個(gè)抽象類(lèi),必須在 class 前面加上 abstract 關(guān)鍵字,并且不能被實(shí)例化。
abstract class A
{
/** 抽象類(lèi)中可以定義變量 */
protected $value1 = 0;
private $value2 = 1;
public $value3 = 2;
/** 也可以定義非抽象方法 */
public function my_print()
{
echo "hello,world/n";
}
/**
* 大多數(shù)情況下,抽象類(lèi)至少含有一個(gè)抽象方法。抽象方法用abstract關(guān)鍵字聲明,其中不能有具體內(nèi)容。
* 可以像聲明普通類(lèi)方法那樣聲明抽象方法,但是要以分號(hào)而不是方法體結(jié)束。也就是說(shuō)抽象方法在抽象類(lèi)中不能被實(shí)現(xiàn),也就是沒(méi)有函數(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";
}
/** 這么寫(xiě)在zend studio 8中會(huì)報(bào)錯(cuò)*/
//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)建了一個(gè)繼承自 A 的子類(lèi) B ,但是不實(shí)現(xiàn)抽象方法 abstract_func() :
Class B extends A{};
那么程序?qū)⒊霈F(xiàn)以下錯(cuò)誤:
Fatal error: Class B contains 1 abstract method and must therefore be declared abstract or implement the remaining methods (A::abstract_func)
5 .如果 B 實(shí)現(xiàn)了抽象方法 abstract_func() ,那么 B 中 abstract_func() 方法的訪問(wèn)控制不能比 A 中 abstract_func() 的訪問(wèn)控制更嚴(yán)格,也就是說(shuō):
(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 .抽象類(lèi)提供了具體實(shí)現(xiàn)的標(biāo)準(zhǔn),而接口則是純粹的模版。接口只定義功能,而不包含實(shí)現(xiàn)的內(nèi)容。接口用關(guān)鍵字 interface 來(lái)聲明。
2 . interface 是完全抽象的,只能聲明方法,而且只能聲明 public 的方法,不能聲明 private 及 protected 的方法,不能定義方法體,也不能聲明實(shí)例變量 。然而, interface 卻可以聲明常量變量 。但將常量變量放在 interface 中違背了其作為接口的作用而存在的宗旨,也混淆了 interface 與類(lèi)的不同價(jià)值。如果的確需要,可以將其放在相應(yīng)的 abstract class 或 Class 中。
interface iA
{
const AVAR=3;
public function iAfunc1();
public function iAfunc2();
}
echo iA:: AVAR;
3 .任何實(shí)現(xiàn)接口的類(lèi)都要實(shí)現(xiàn)接口中所定義的所有方法
class E implements iA
{
public function iAfunc1(){echo "in iAfunc1";}
public function iAfunc2(){echo "in iAfunc2";}
}
否則該類(lèi)必須聲明為 abstract 。
abstract class E implements iA{}
4 .一個(gè)類(lèi)可以在聲明中使用 implements 關(guān)鍵字來(lái)實(shí)現(xiàn)某個(gè)接口。這么做之后,實(shí)現(xiàn)接口的具體過(guò)程和繼承一個(gè)僅包含抽象方法的抽象類(lèi)是一樣的。一個(gè)類(lèi)可以同時(shí)繼承一個(gè)父類(lèi)和實(shí)現(xiàn)任意多個(gè)接口。 extends 子句應(yīng)該在 implements 子句之前。 PHP 只支持繼承自一個(gè)父類(lèi),因此 extends 關(guān)鍵字后只能跟一個(gè)類(lèi)名。
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 .接口不可以實(shí)現(xiàn)另一個(gè)接口,但可以繼承多個(gè)
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";}
}
三、 抽象類(lèi)和接口的異同
1. 相同點(diǎn):
(1) 兩者都是抽象類(lèi),都不能實(shí)例化。
(2) interface 實(shí)現(xiàn)類(lèi)及 abstract class 的子類(lèi)都必須要實(shí)現(xiàn)已經(jīng)聲明的抽象方法。
2. 不同點(diǎn):
(1) interface 需要實(shí)現(xiàn),要用 implements ,而 abstract class 需要繼承,要用 extends 。
(2) 一個(gè)類(lèi)可以實(shí)現(xiàn)多個(gè) interface ,但一個(gè)類(lèi)只能繼承一個(gè) abstract class 。
(3) interface 強(qiáng)調(diào)特定功能的實(shí)現(xiàn),而 abstract class 強(qiáng)調(diào)所屬關(guān)系。
(4) 盡管 interface 實(shí)現(xiàn)類(lèi)及 abstract class 的子類(lèi)都必須要實(shí)現(xiàn)相應(yīng)的抽象方法,但實(shí)現(xiàn)的形式不同。 interface 中的每一個(gè)方法都是抽象方法,都只是聲明的 (declaration, 沒(méi)有方法體 ) ,實(shí)現(xiàn)類(lèi)必須要實(shí)現(xiàn)。而 abstract class 的子類(lèi)可以有選擇地實(shí)現(xiàn)。這個(gè)選擇有兩點(diǎn)含義: a) abstract class 中并非所有的方法都是抽象的,只有那些冠有 abstract 的方法才是抽象的,子類(lèi)必須實(shí)現(xiàn)。那些沒(méi)有 abstract 的方法,在 abstract class 中必須定義方法體; b) abstract class 的子類(lèi)在繼承它時(shí),對(duì)非抽象方法既可以直接繼承,也可以覆蓋;而對(duì)抽象方法,可以選擇實(shí)現(xiàn),也可以留給其子類(lèi)來(lái)實(shí)現(xiàn),但此類(lèi)必須也聲明為抽象類(lèi)。既是抽象類(lèi),當(dāng)然也不能實(shí)例化。
(5) abstract class 是 interface 與 class 的中介。 abstract class 在 interface 及 class 中起到了承上啟下的作用。一方面, abstract class 是抽象的,可以聲明抽象方法,以規(guī)范子類(lèi)必須實(shí)現(xiàn)的功能;另一方面,它又可以定義缺省的方法體,供子類(lèi)直接使用或覆蓋。另外,它還可以定義自己的實(shí)例變量,以供子類(lèi)通過(guò)繼承來(lái)使用。
(6) 接口中的抽象方法前不用也不能加 abstract 關(guān)鍵字,默認(rèn)隱式就是抽象方法,也不能加 final 關(guān)鍵字來(lái)防止抽象方法的繼承。而抽象類(lèi)中抽象方法前則必須加上 abstract 表示顯示聲明為抽象方法。
(7) 接口中的抽象方法默認(rèn)是 public 的,也只能是 public 的,不能用 private , protected 修飾符修飾。而抽象類(lèi)中的抽象方法則可以用 public , protected 來(lái)修飾,但不能用 private 。
3. interface 的應(yīng)用場(chǎng)合
(1) 類(lèi)與類(lèi)之間需要特定的接口進(jìn)行協(xié)調(diào),而不在乎其如何實(shí)現(xiàn)。
(2) 作為能夠?qū)崿F(xiàn)特定功能的標(biāo)識(shí)存在,也可以是什么接口方法都沒(méi)有的純粹標(biāo)識(shí)。
(3) 需要將一組類(lèi)視為單一的類(lèi),而調(diào)用者只通過(guò)接口來(lái)與這組類(lèi)發(fā)生聯(lián)系。
(4) 需要實(shí)現(xiàn)特定的多項(xiàng)功能,而這些功能之間可能完全沒(méi)有任何聯(lián)系。
4. abstract class 的應(yīng)用場(chǎng)合
一句話,在既需要統(tǒng)一的接口,又需要實(shí)例變量或缺省的方法的情況下,就可以使用它。最常見(jiàn)的有:
(1) 定義了一組接口,但又不想強(qiáng)迫每個(gè)實(shí)現(xiàn)類(lèi)都必須實(shí)現(xiàn)所有的接口??梢杂?abstract class 定義一組方法體,甚至可以是空方法體,然后由子類(lèi)選擇自己所感興趣的方法來(lái)覆蓋。
(2) 某些場(chǎng)合下,只靠純粹的接口不能滿足類(lèi)與類(lèi)之間的協(xié)調(diào),還必需類(lèi)中表示狀態(tài)的變量來(lái)區(qū)別不同的關(guān)系。 abstract 的中介作用可以很好地滿足這一點(diǎn)。
(3) 規(guī)范了一組相互協(xié)調(diào)的方法,其中一些方法是共同的,與狀態(tài)無(wú)關(guān)的,可以共享的,無(wú)需子類(lèi)分別實(shí)現(xiàn);而另一些方法卻需要各個(gè)子類(lèi)根據(jù)自己特定的狀態(tài)來(lái)實(shí)現(xiàn)特 定的功能 。
- PHP面向?qū)ο笪宕笤瓌t之接口隔離原則(ISP)詳解
- php 接口類(lèi)與抽象類(lèi)的實(shí)際作用
- 詳細(xì)解讀PHP中接口的應(yīng)用
- 淺談php處理后端&接口訪問(wèn)超時(shí)的解決方法
- 領(lǐng)悟php接口中interface存在的意義
- PHP 的ArrayAccess接口 像數(shù)組一樣來(lái)訪問(wèn)你的PHP對(duì)象
- PHP微信API接口類(lèi)
- 深入分析php中接口與抽象類(lèi)的區(qū)別
- PHP接口并發(fā)測(cè)試的方法(推薦)
- php接口與接口引用的深入解析
- php面向?qū)ο笕ヂ?(十四) php5接口技術(shù)
- php接口隔離原則實(shí)例分析
相關(guān)文章
ThinkPHP實(shí)現(xiàn)更新數(shù)據(jù)實(shí)例詳解(demo)
本文給大家介紹thinkphp實(shí)現(xiàn)更新數(shù)據(jù)的實(shí)例詳解以及thinkphp更新數(shù)據(jù)庫(kù)的五種方法,本文介紹的非常不錯(cuò),具有參考借鑒價(jià)值,感興趣的朋友可以參考下2016-06-06IIS下配置頁(yè)面重寫(xiě)(配合插件url-rewrite2去除頁(yè)面后綴名)的實(shí)現(xiàn)方法
這篇文章主要介紹了IIS下配置頁(yè)面重寫(xiě)(配合插件url-rewrite2去除頁(yè)面后綴名)的實(shí)現(xiàn)方法,需要的朋友可以參考下2017-10-10Thinkphp 框架擴(kuò)展之應(yīng)用模式實(shí)現(xiàn)方法分析
這篇文章主要介紹了Thinkphp 框架擴(kuò)展之應(yīng)用模式,結(jié)合實(shí)例形式分析了Thinkphp 擴(kuò)展應(yīng)用模式的具體原理、實(shí)現(xiàn)方法及相關(guān)注意事項(xiàng),需要的朋友可以參考下2020-04-04