PHP 中的抽象類(abstract class)和介面(interface)

jiazhuamh發表於2017-05-05

一、 抽象類abstract class

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

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

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

    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() :

Class B extends A{};

程式會報錯。

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 中。

    interface iA  
    {  
        const AVAR=3;  
        public function iAfunc1();  
        public function iAfunc2();  
    }  
    echo iA:: AVAR;  

3 .任何實現介面的類都要實現介面中所定義的所有方法

    class E implements iA  
    {  
        public function iAfunc1(){echo "in iAfunc1";}  
        public function iAfunc2(){echo "in iAfunc2";}  
    }  

否則該類必須宣告為 abstract 。

    abstract class E implements iA{}  

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

    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 .介面不可以實現另一個介面,但可以繼承多個

    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 的子類都必須要實現已經宣告的抽象方法。

  1. 不同點:

(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 。

  1. interface 的應用場合

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

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

(3) 需要將一組類視為單一的類,而呼叫者只通過介面來與這組類發生聯絡。

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

  1. abstract class 的應用場合

一句話,在既需要統一的介面,又需要例項變數或預設的方法的情況下,就可以使用它。最常見的有:

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

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

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

本作品採用《CC 協議》,轉載必須註明作者和本文連結

相關文章