區(qū)別為:1、接口是通過(guò)interface關(guān)鍵字來(lái)定義的,抽象類是通過(guò)abstract關(guān)鍵字來(lái)定義的;2、接口沒(méi)有數(shù)據(jù)成員,但是抽象類有數(shù)據(jù)成員,抽象類可以實(shí)現(xiàn)數(shù)據(jù)的封裝;3、接口沒(méi)有構(gòu)造函數(shù),抽象類可以有構(gòu)造函數(shù)。
本教程操作環(huán)境:windows7系統(tǒng)、PHP7.1版,DELL G3電腦
1、抽象類與接口的區(qū)別
在學(xué)習(xí)PHP面向?qū)ο髸r(shí),都會(huì)在抽象類與接口上迷惑,作用差不多為什么還那么容易混淆,何不留一去一?但是事實(shí)上兩者的區(qū)別還是很大的,如果能夠很好地運(yùn)用PHP的兩個(gè)方法,面向?qū)ο蟮某绦蛟O(shè)計(jì)將會(huì)更加合理、清晰高效。
a.接口是通過(guò) interface 關(guān)鍵字來(lái)定義的, 抽象類是通過(guò)abstract關(guān)鍵字來(lái)定義的。
b.對(duì)接口的使用方式是通過(guò)關(guān)鍵字implements來(lái)實(shí)現(xiàn)的,而對(duì)于抽象類的操作是使用類繼承的關(guān)鍵字extends實(shí)現(xiàn)的,使用時(shí)要特別注意。
c.接口沒(méi)有數(shù)據(jù)成員,但是抽象類有數(shù)據(jù)成員,抽象類可以實(shí)現(xiàn)數(shù)據(jù)的封裝。
d.接口沒(méi)有構(gòu)造函數(shù),抽象類可以有構(gòu)造函數(shù)。
e.接口中的方法都是public類型,而抽象類中的方法可以使用private、protected或public來(lái)修飾。
f.一個(gè)類可以同時(shí)實(shí)現(xiàn)多個(gè)接口,但是只能實(shí)現(xiàn)一個(gè)抽象類。
相同點(diǎn):抽象方法與接口的函數(shù)體內(nèi)不能寫任何東西,連兩個(gè)大括號(hào)都不能寫!?。∪纾篺unction getName();這樣就行了
2、接口
使用接口(interface),可以指定某個(gè)類必須實(shí)現(xiàn)哪些方法,但不需要定義這些方法的具體內(nèi)容。
接口是通過(guò) interface 關(guān)鍵字來(lái)定義的,就像定義一個(gè)標(biāo)準(zhǔn)的類一樣,但其中定義所有的方法都是空的。
接口中定義的所有方法都必須是公有,這是接口的特性。
實(shí)現(xiàn)(implements)
要實(shí)現(xiàn)一個(gè)接口,使用 implements 操作符。類中必須實(shí)現(xiàn)接口中定義的所有方法,否則會(huì)報(bào)一個(gè)致命錯(cuò)誤。類可以實(shí)現(xiàn)多個(gè)接口,用逗號(hào)來(lái)分隔多個(gè)接口的名稱。
Note:
實(shí)現(xiàn)多個(gè)接口時(shí),接口中的方法不能有重名。
Note:
接口也可以繼承,通過(guò)使用 extends 操作符。
Note:
類要實(shí)現(xiàn)接口,必須使用和接口中所定義的方法完全一致的方式。否則會(huì)導(dǎo)致致命錯(cuò)誤。
常量
接口中也可以定義常量。接口常量和類常量的使用完全相同,但是不能被子類或子接口所覆蓋。
<?php // 聲明一個(gè)'iTemplate'接口 interface iTemplate { public function setVariable($name, $var); public function getHtml($template); } // 實(shí)現(xiàn)接口 // 下面的寫法是正確的 class Template implements iTemplate { private $vars = array(); public function setVariable($name, $var) { $this->vars[$name] = $var; } public function getHtml($template) { foreach($this->vars as $name => $value) { $template = str_replace('{' . $name . '}', $value, $template); } return $template; } } // 下面的寫法是錯(cuò)誤的,會(huì)報(bào)錯(cuò),因?yàn)闆](méi)有實(shí)現(xiàn) getHtml(): // Fatal error: Class BadTemplate contains 1 abstract methods // and must therefore be declared abstract (iTemplate::getHtml) class BadTemplate implements iTemplate { private $vars = array(); public function setVariable($name, $var) { $this->vars[$name] = $var; } } ?> Example #2 可擴(kuò)充的接口 <?php interface a { public function foo(); } interface b extends a { public function baz(Baz $baz); } // 正確寫法 class c implements b { public function foo() { } public function baz(Baz $baz) { } } // 錯(cuò)誤寫法會(huì)導(dǎo)致一個(gè)致命錯(cuò)誤 class d implements b { public function foo() { } public function baz(Foo $foo) { } } ?> Example #3 繼承多個(gè)接口 <?php interface a { public function foo(); } interface b { public function bar(); } interface c extends a, b { public function baz(); } class d implements c { public function foo() { } public function bar() { } public function baz() { } } ?> Example #4 使用接口常量 <?php interface a { const b = 'Interface constant'; } // 輸出接口常量 echo a::b; // 錯(cuò)誤寫法,因?yàn)槌A坎荒鼙桓采w。接口常量的概念和類常量是一樣的。 class b implements a { const b = 'Class constant'; } ?>
http://php.net/manual/zh/language.oop5.interfaces.php
3、抽象類
PHP 5 支持抽象類和抽象方法。定義為抽象的類不能被實(shí)例化。任何一個(gè)類,如果它里面至少有一個(gè)方法是被聲明為抽象的,那么這個(gè)類就必須被聲明為抽象的。被定義為抽象的方法只是聲明了其調(diào)用方式(參數(shù)),不能定義其具體的功能實(shí)現(xiàn)。
繼承一個(gè)抽象類的時(shí)候,子類必須定義父類中的所有抽象方法;另外,這些方法的訪問(wèn)控制必須和父類中一樣(或者更為寬松)。例如某個(gè)抽象方法被聲明為受保護(hù)的,那么子類中實(shí)現(xiàn)的方法就應(yīng)該聲明為受保護(hù)的或者公有的,而不能定義為私有的。此外方法的調(diào)用方式必須匹配,即類型和所需參數(shù)數(shù)量必須一致。例如,子類定義了一個(gè)可選參數(shù),而父類抽象方法的聲明里沒(méi)有,則兩者的聲明并無(wú)沖突。 這也適用于 PHP 5.4 起的構(gòu)造函數(shù)。在 PHP 5.4 之前的構(gòu)造函數(shù)聲明可以不一樣的。
<?php abstract class AbstractClass { // 強(qiáng)制要求子類定義這些方法 abstract protected function getValue(); abstract protected function prefixValue($prefix); // 普通方法(非抽象方法) public function printOut() { print $this->getValue() . "n"; } } class ConcreteClass1 extends AbstractClass { protected function getValue() { return "ConcreteClass1"; } public function prefixValue($prefix) { return "{$prefix}ConcreteClass1"; } } class ConcreteClass2 extends AbstractClass { public function getValue() { return "ConcreteClass2"; } public function prefixValue($prefix) { return "{$prefix}ConcreteClass2"; } } $class1 = new ConcreteClass1; $class1->printOut(); echo $class1->prefixValue('FOO_') ."n"; $class2 = new ConcreteClass2; $class2->printOut(); echo $class2->prefixValue('FOO_') ."n"; ?> 以上例程會(huì)輸出: ConcreteClass1 FOO_ConcreteClass1 ConcreteClass2 FOO_ConcreteClass2 Example #2 抽象類示例 <?php abstract class AbstractClass { // 我們的抽象方法僅需要定義需要的參數(shù) abstract protected function prefixName($name); } class ConcreteClass extends AbstractClass { // 我們的子類可以定義父類簽名中不存在的可選參數(shù) public function prefixName($name, $separator = ".") { if ($name == "Pacman") { $prefix = "Mr"; } elseif ($name == "Pacwoman") { $prefix = "Mrs"; } else { $prefix = ""; } return "{$prefix}{$separator} {$name}"; } } $class = new ConcreteClass; echo $class->prefixName("Pacman"), "n"; echo $class->prefixName("Pacwoman"), "n"; ?> 以上例程會(huì)輸出: Mr. Pacman Mrs. Pacwoman 老代碼中如果沒(méi)有自定義類或函數(shù)被命名為“abstract”,則應(yīng)該能不加修改地正常運(yùn)行。
http://php.net/manual/zh/language.oop5.abstract.php
推薦學(xué)習(xí):《PHP視頻教程》