php抽象類和接口的區別
tags:抽象類 接口 抽象類和接口 php
引言:這是一個面試經常被問到的問題,也是一個經典問題。我們盡量引用官方權威的說明或者經過實驗來證明本文所說的內容准確性。
抽象類
官方描述請查看文檔,下面是官方描述的梳理版本:
- 定義為抽象的類不能被實例化。任何一個類,如果它里面至少有一個方法是被聲明為抽象的,那么這個類就必須被聲明為抽象的。(抽象類可以沒有抽象方法,但是抽象類依然不能被實例化)被定義為抽象的方法只是聲明了其調用方式(參數),不能定義其具體的功能實現。 如
abstract class AbstractClass
{
// 強制要求子類定義這些方法,且不能定義具體功能 注意沒有大括號{}
abstract protected function getValue ();
abstract protected function prefixValue ( $prefix );
// 普通方法(非抽象方法)
public function printOut () {
print $this -> getValue () . "\n" ;
}
}
- 繼承一個抽象類的時候,非抽象子類必須定義父類中的所有抽象方法;另外,這些方法的訪問控制必須和父類中一樣(或者更為寬松)。例如某個抽象方法被聲明為受保護的,那么子類中實現的方法就應該聲明為受保護的或者公有的,而不能定義為私有的。
- 此外方法的調用方式必須匹配,即類型和所需參數數量必須一致。例如,子類定義了一個可選參數(類似function eat($a,$b=1)中的$b就是可選參數),而父類抽象方法的聲明里沒有,則兩者的聲明並無沖突。這也適用於 PHP 5.4 起的構造函數。在 PHP 5.4 之前的構造函數聲明可以不一樣的。
補充:
- 抽象類可以有成員屬性。
- 有人問:抽象方法是否可以定義為私有,答案是不可以,因為抽象接口的目的就是為了抽象出類模型用來繼承,定義為私有,外部訪問不到,偏移了設計目的。如下會報錯
- 抽象類可以實現接口,且可以不實現其中的方法(后面接口的實現中有代碼)
```php
abstract class Sutdent extends Human
{
abstract private function study();
}
```
Fatal error: Abstract function Sutdent::study() cannot be declared private in D:\11\index.php on line 10
- 抽象類可以繼承抽象類,且不能重寫抽象父類的抽象方法。這樣的用法,可以理解為對抽象類的擴展。如
abstract class Human
{
abstract function eat();
}
abstract class Sutdent extends Human
{
abstract function study();
//abstract function eat(); 若重寫抽象父類的抽象方法eat()會報錯
}
若重寫抽象父類的抽象方法則報以下錯誤
Fatal error: Can't inherit abstract function Human::eat() (previously declared abstract in Sutdent) in D:\11\index.php on line 11
接口
1. 接口的定義
- 使用接口(interface),可以指定某個類必須實現哪些方法,但不需要定義這些方法的具體內容。
- 接口是通過 interface 關鍵字來定義的,就像定義一個標准的類一樣,但其中定義所有的方法都是空的。
- 接口中定義的所有方法都必須是公有,這是接口的特性,protected和private會報錯(Fatal error: Access type for interface method)。
- 常量:接口中也可以定義常量。接口常量和類常量的使用完全相同,但是不能被子類或子接口所覆蓋。(不建議這樣用,實在想不到有什么意義,也容易產生和抽象類的混淆)
interface Play
{
const LEVEL=10;
public function PlayLOL();
public function PlayFootball();
}
2. 接口的實現
要實現一個接口,使用 implements 操作符。非抽象類中必須實現接口中定義的所有方法,否則會報一個致命錯誤。類可以實現多個接口,用逗號來分隔多個接口的名稱。
補充:
- 可以同時繼承抽象類和實現接口,extends要寫在前面.
- 抽象類實現接口,不需要重新其中的方法。
- 實現多個接口時,接口中的方法不能有重名。
- 接口也可以繼承,通過使用 extends 操作符。
- 類要實現接口,必須使用和接口中所定義的方法完全一致的方式。否則會導致致命錯誤。
interface Play
{
const LEVEL=10;
public function PlayLOL();
public function PlayFootball();
}
interface Read
{
public function ReadNovel();
}
abstract class Human
{
abstract function eat();
}
//抽象類可以實現接口后不實現其方法,可以繼承一個抽象類的同時實現多個接口注意必須要把extends語句寫在implements前面,否則會報錯
abstract class Sutdent extends Human implements Play,Read
{
abstract function study();
}
3. 接口的繼承
接口可以繼承另一個或多個接口,使用extends關鍵字,多個用 ',' 隔開,但是不能實現另一個接口,當然更不能繼承抽象類(繼承抽象類報錯:Fatal error: PlayGame cannot implement Human - it is not an interface in D:\11\index.php on line 10)
interface Play
{
public function PlayFootball();
}
interface PlayNew
{
public function PlayFootballNew();
}
interface PlayGame extends play,PlayNew
{
public function PlayLOL();
}
總結
一般都在這里寫相同點和不同點,我偏不寫,嘿嘿,因為我覺得上面寫的夠詳細了。
我們簡單總結一句:抽象類一般用來定義一類實體是什么,他包含了屬性,抽象方法和非抽象方法。接口用來定義一類實體能做什么,一般認為他只有抽象方法,常量極少用到。