本文範例講述了PHP中abstract(抽象)、final(最終)和static(靜態)原理與用法。分享給大家供大家參考,具體如下:
PHP 5 支援抽象類和抽象方法。定義為抽象的類不能被範例化。任何一個類,如果它裡面至少有一個方法是被宣告為抽象的,那麼這個類就必須被宣告為抽象的。被定義為抽象的方法只是宣告了其呼叫方式(引數),不能定義其具體的功能實現。
繼承一個抽象類的時候,子類必須定義父類別中的所有抽象方法;另外,這些方法的存取控制必須和父類別中一樣(或者更為寬鬆)。例如某個抽象方法被宣告為受保護的,那麼子類中實現的方法就應該宣告為受保護的或者公有的(嚴格程度:private>protected>public),而不能定義為私有的。此外方法的呼叫方式必須匹配,即型別和所需引數數量必須一致。例如,子類定義了一個可選引數,而父類別抽象方法的宣告裡沒有,則兩者的宣告並無衝突。 這也適用於 PHP 5.4 起的建構函式。在 PHP 5.4 之前的建構函式宣告可以不一樣的。
總結:
抽象類不能被範例化;
類中有任何抽象方法那這個類也必須為抽象的;
抽象類只能申明呼叫方式和引數,不能定義具體功能實現;
繼承抽象類的子類必須實現抽象類的所有抽象方法;
子類中實現的抽象方法的存取控制必須比父類別的存取控制更嚴格;
子類中實現的方法的呼叫方式及引數數量必須與被實現的方法一致。
例:
<?php abstract class AbstractClass { // 強制要求子類定義這些方法,不定義功能實現 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"; ?>
<?php abstract class AbstractClass { // 我們的抽象方法僅需要定義需要的引數 abstract protected function prefixName($name); } class ConcreteClass extends AbstractClass { // 我們的子類可以定義父類別簽名中不存在的 可選引數 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"; ?>
如果父類別中的方法被宣告為 final,則子類無法覆蓋該方法。如果一個類被宣告為 final,則不能被繼承。
這個比較好理解,不做贅述
宣告類屬性或方法為靜態,就可以不範例化類而直接存取。靜態屬性不能通過一個類已範例化的物件來存取(但靜態方法可以)。
為了相容 PHP 4,如果沒有指定存取控制,屬性和方法預設為公有。
由於靜態方法不需要通過物件即可呼叫,所以偽變數 $this
在靜態方法中不可用。
靜態屬性不可以由物件通過 ->
操作符來存取。
用靜態方式呼叫一個非靜態方法會導致一個 E_STRICT
級別的錯誤。
就像其它所有的 PHP 靜態變數一樣,靜態屬性只能被初始化為文字或常數,不能使用表示式。所以可以把靜態屬性初始化為整數或陣列,但不能初始化為另一個變數或函數返回值,也不能指向一個物件。
自 PHP 5.3.0 起,可以用一個變數來動態呼叫類。但該變數的值不能為關鍵字 self,parent 或 static。
總結:
靜態方法無需範例化,可直接存取;
類範例化的物件無法存取類中的靜態屬性,但是可以存取靜態方法;
偽變數 $this
在靜態方法中不可用;
靜態屬性不可以由物件通過 -> 操作符來存取;
用靜態方式呼叫一個非靜態方法會導致一個 E_STRICT
級別的錯誤;
靜態屬性只能被初始化為文字或常數,不能使用表示式(函數返回值/寧一個變數/物件);
可以用一個變數來動態呼叫類。但該變數的值不能為關鍵字 self,parent 或 static。
<?php class Foo { public static $my_static = 'foo'; public function staticValue() { return self::$my_static; } } class Bar extends Foo { public function fooStatic() { return parent::$my_static; } } print Foo::$my_static . "n"; $foo = new Foo(); print $foo->staticValue() . "n"; print $foo->my_static . "n"; // Undefined "Property" my_static print $foo::$my_static . "n"; $classname = 'Foo'; print $classname::$my_static . "n"; // As of PHP 5.3.0 print Bar::$my_static . "n"; $bar = new Bar(); print $bar->fooStatic() . "n"; ?> </programlisting> </example> <example> <title>靜態方法範例</title> <programlisting role="php"> <![CDATA[ <?php class Foo { public static function aStaticMethod() { // ... } } Foo::aStaticMethod(); $classname = 'Foo'; $classname::aStaticMethod(); // 自 PHP 5.3.0 起 ?>
以上就是詳解PHP中abstract(抽象)、final(最終)和static(靜態)原理與用法的詳細內容,更多請關注TW511.COM其它相關文章!