本文實(shí)例講述了PHP中abstract(抽象)、final(最終)和static(靜態(tài))原理與用法。分享給大家供大家參考,具體如下:
abstract(抽象)
PHP 5 支持抽象類(lèi)和抽象方法。定義為抽象的類(lèi)不能被實(shí)例化。任何一個(gè)類(lèi),如果它里面至少有一個(gè)方法是被聲明為抽象的,那么這個(gè)類(lèi)就必須被聲明為抽象的。被定義為抽象的方法只是聲明了其調(diào)用方式(參數(shù)),不能定義其具體的功能實(shí)現(xiàn)。
繼承一個(gè)抽象類(lèi)的時(shí)候,子類(lèi)必須定義父類(lèi)中的所有抽象方法;另外,這些方法的訪問(wèn)控制必須和父類(lèi)中一樣(或者更為寬松)。例如某個(gè)抽象方法被聲明為受保護(hù)的,那么子類(lèi)中實(shí)現(xiàn)的方法就應(yīng)該聲明為受保護(hù)的或者公有的(嚴(yán)格程度:private>protected>public),而不能定義為私有的。此外方法的調(diào)用方式必須匹配,即類(lèi)型和所需參數(shù)數(shù)量必須一致。例如,子類(lèi)定義了一個(gè)可選參數(shù),而父類(lèi)抽象方法的聲明里沒(méi)有,則兩者的聲明并無(wú)沖突。 這也適用于 PHP 5.4 起的構(gòu)造函數(shù)。在 PHP 5.4 之前的構(gòu)造函數(shù)聲明可以不一樣的。
總結(jié):
- 抽象類(lèi)不能被實(shí)例化;
- 類(lèi)中有任何抽象方法那這個(gè)類(lèi)也必須為抽象的;
- 抽象類(lèi)只能申明調(diào)用方式和參數(shù),不能定義具體功能實(shí)現(xiàn);
- 繼承抽象類(lèi)的子類(lèi)必須實(shí)現(xiàn)抽象類(lèi)的所有抽象方法;
- 子類(lèi)中實(shí)現(xiàn)的抽象方法的訪問(wèn)控制必須比父類(lèi)的訪問(wèn)控制更嚴(yán)格;
- 子類(lèi)中實(shí)現(xiàn)的方法的調(diào)用方式及參數(shù)數(shù)量必須與被實(shí)現(xiàn)的方法一致。
例:
?php
abstract class AbstractClass
{
// 強(qiáng)制要求子類(lèi)定義這些方法,不定義功能實(shí)現(xiàn)
abstract protected function getValue();
abstract protected function prefixValue($prefix);
// 普通方法(非抽象方法),子類(lèi)可以不重寫(xiě)
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
{
//訪問(wèn)方式可以更寬松
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
{
// 我們的抽象方法僅需要定義需要的參數(shù)
abstract protected function prefixName($name);
}
class ConcreteClass extends AbstractClass
{
// 我們的子類(lèi)可以定義父類(lèi)簽名中不存在的 可選參數(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";
?>
final
如果父類(lèi)中的方法被聲明為 final,則子類(lèi)無(wú)法覆蓋該方法。如果一個(gè)類(lèi)被聲明為 final,則不能被繼承。
這個(gè)比較好理解,不做贅述
static
聲明類(lèi)屬性或方法為靜態(tài),就可以不實(shí)例化類(lèi)而直接訪問(wèn)。靜態(tài)屬性不能通過(guò)一個(gè)類(lèi)已實(shí)例化的對(duì)象來(lái)訪問(wèn)(但靜態(tài)方法可以)。
為了兼容 PHP 4,如果沒(méi)有指定訪問(wèn)控制,屬性和方法默認(rèn)為公有。
由于靜態(tài)方法不需要通過(guò)對(duì)象即可調(diào)用,所以偽變量 $this 在靜態(tài)方法中不可用。
靜態(tài)屬性不可以由對(duì)象通過(guò) -> 操作符來(lái)訪問(wèn)。
用靜態(tài)方式調(diào)用一個(gè)非靜態(tài)方法會(huì)導(dǎo)致一個(gè) E_STRICT 級(jí)別的錯(cuò)誤。
就像其它所有的 PHP 靜態(tài)變量一樣,靜態(tài)屬性只能被初始化為文字或常量,不能使用表達(dá)式。所以可以把靜態(tài)屬性初始化為整數(shù)或數(shù)組,但不能初始化為另一個(gè)變量或函數(shù)返回值,也不能指向一個(gè)對(duì)象。
自 PHP 5.3.0 起,可以用一個(gè)變量來(lái)動(dòng)態(tài)調(diào)用類(lèi)。但該變量的值不能為關(guān)鍵字 self,parent 或 static。
總結(jié):
- 靜態(tài)方法無(wú)需實(shí)例化,可直接訪問(wèn);
- 類(lèi)實(shí)例化的對(duì)象無(wú)法訪問(wèn)類(lèi)中的靜態(tài)屬性,但是可以訪問(wèn)靜態(tài)方法;
- 偽變量 $this 在靜態(tài)方法中不可用;
- 靜態(tài)屬性不可以由對(duì)象通過(guò) -> 操作符來(lái)訪問(wèn);
- 用靜態(tài)方式調(diào)用一個(gè)非靜態(tài)方法會(huì)導(dǎo)致一個(gè) E_STRICT 級(jí)別的錯(cuò)誤;
- 靜態(tài)屬性只能被初始化為文字或常量,不能使用表達(dá)式(函數(shù)返回值/寧一個(gè)變量/對(duì)象);
- 可以用一個(gè)變量來(lái)動(dòng)態(tài)調(diào)用類(lèi)。但該變量的值不能為關(guān)鍵字 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>靜態(tài)方法示例/title>
programlisting role="php">
![CDATA[
?php
class Foo {
public static function aStaticMethod() {
// ...
}
}
Foo::aStaticMethod();
$classname = 'Foo';
$classname::aStaticMethod(); // 自 PHP 5.3.0 起
?>
更多關(guān)于PHP相關(guān)內(nèi)容感興趣的讀者可查看本站專(zhuān)題:《php面向?qū)ο蟪绦蛟O(shè)計(jì)入門(mén)教程》、《PHP數(shù)組(Array)操作技巧大全》、《PHP基本語(yǔ)法入門(mén)教程》、《PHP運(yùn)算與運(yùn)算符用法總結(jié)》、《php字符串(string)用法總結(jié)》、《php+mysql數(shù)據(jù)庫(kù)操作入門(mén)教程》及《php常見(jiàn)數(shù)據(jù)庫(kù)操作技巧匯總》
希望本文所述對(duì)大家PHP程序設(shè)計(jì)有所幫助。
您可能感興趣的文章:- Django User 模塊之 AbstractUser 擴(kuò)展詳解
- JavaSE static final及abstract修飾符實(shí)例解析
- Python2和Python3中@abstractmethod使用方法
- JAVA抽象類(lèi)和抽象方法(abstract)實(shí)例分析
- Springboot源碼 AbstractAdvisorAutoProxyCreator解析
- java編程abstract類(lèi)和方法詳解
- 淺談利用Spring的AbstractRoutingDataSource解決多數(shù)據(jù)源的問(wèn)題
- Django如何繼承AbstractUser擴(kuò)展字段