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

1、 抽象類abstract classphp

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

2 .抽象類不能被直接實例化。抽象類中只定義(或部分實現)子類須要的方法。子類能夠經過繼承抽象類並經過實現抽象類中的全部抽象方法,使抽象類具體化。函數

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

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
abstract  class 
     /** 抽象類中能夠定義變量 */ 
     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 
     public  function  abstract_func1() 
    
        echo  "implement the abstract_func1 in class A/n"
    
     /** 這麼寫在zend studio 8中會報錯*/ 
     //abstract protected function abstract_func2(); 
class  C extends 
     public  function  abstract_func2() 
    
        echo  "implement the abstract_func2 in class A/n"
    

 

4 .若是像下面這樣建立了一個繼承自 A 的子類 B ,可是不實現抽象方法 abstract_func() :code

1
Class B extends  A{};

 那麼程序將出現如下錯誤:blog

1
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 ,但不能是 privateci

(3)      若是 A 中 abstract_func() 聲明爲 private ,嘿嘿,不能定義爲 private 哦!( Fatal error : Abstract function A::abstract_func() cannot be declared private )rem

2、 接口interface

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

2 . interface 是徹底抽象的,只能聲明方法,並且只能聲明 public 的方法,不能聲明 private 及 protected 的方法,不能定義方法體,也不能聲明實例變量 。然而, interface 卻能夠聲明常量變量 。但將常量變量放在 interface 中違背了其做爲接口的做用而存在的宗旨,也混淆了 interface 與類的不一樣價值。若是的確須要,能夠將其放在相應的 abstract class 或 Class 中。

1
2
3
4
5
6
7
interface  iA 
     const  AVAR=3; 
     public  function  iAfunc1(); 
     public  function  iAfunc2(); 
echo  iA:: AVAR;

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

1
2
3
4
5
class  E implements  iA 
     public  function  iAfunc1(){ echo  "in iAfunc1" ;} 
     public  function  iAfunc2(){ echo  "in iAfunc2" ;} 

 不然該類必須聲明爲 abstract 。

1
abstract  class  E implements  iA{} 

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

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
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 .接口不能夠實現另外一個接口,但能夠繼承多個

1
2
3
4
5
6
7
8
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" ;} 

 3、 抽象類和接口的異同

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)      規範了一組相互協調的方法,其中一些方法是共同的,與狀態無關的,能夠共享的,無需子類分別實現;而另外一些方法卻須要各個子類根據本身特定的狀態來實現特 定的功能 。

相關文章
相關標籤/搜索