PHP 增加了一個(gè)叫做后期靜態(tài)綁定的功能,用于在繼承范圍內引用靜態(tài)調用的類(lèi)。
準確說(shuō),后期靜態(tài)綁定工作原理是存儲了在上一個(gè)“非轉發(fā)調用”(non-forwarding
call)的類(lèi)名。當進(jìn)行靜態(tài)方法調用時(shí),該類(lèi)名即為明確指定的那個(gè)(通常在
::
運算符左側部分);當進(jìn)行非靜態(tài)方法調用時(shí),即為該對象所屬的類(lèi)。所謂的“轉發(fā)調用”(forwarding
call)指的是通過(guò)以下幾種方式進(jìn)行的靜態(tài)調用:self::
,parent::
,static::
以及 forward_static_call()??捎?get_called_class()
函數來(lái)得到被調用的方法所在的類(lèi)名,static::
則指出了其范圍。
該功能從語(yǔ)言?xún)炔拷嵌瓤紤]被命名為“后期靜態(tài)綁定”?!昂笃诮壎ā钡囊馑际钦f(shuō),static::
不再被解析為定義當前方法所在的類(lèi),而是在實(shí)際運行時(shí)計算的。也可以稱(chēng)之為“靜態(tài)綁定”,因為它可以用于(但不限于)靜態(tài)方法的調用。
self::
的限制
使用 self::
或者
__CLASS__
對當前類(lèi)的靜態(tài)引用,取決于定義當前方法所在的類(lèi):
示例 #1 self::
用法
<?php
class A {
public static function who() {
echo __CLASS__;
}
public static function test() {
self::who();
}
}
class B extends A {
public static function who() {
echo __CLASS__;
}
}
B::test();
?>
以上例程會(huì )輸出:
A
后期靜態(tài)綁定本想通過(guò)引入一個(gè)新的關(guān)鍵字表示運行時(shí)最初調用的類(lèi)來(lái)繞過(guò)限制。簡(jiǎn)單地說(shuō),這個(gè)關(guān)鍵字能在上述例子中調用
test()
時(shí)引用的類(lèi)是 B
而不是
A
。最終決定不引入新的關(guān)鍵字,而是使用已經(jīng)預留的 static
關(guān)鍵字。
示例 #2 static::
簡(jiǎn)單用法
<?php
class A {
public static function who() {
echo __CLASS__;
}
public static function test() {
static::who(); // 后期靜態(tài)綁定從這里開(kāi)始
}
}
class B extends A {
public static function who() {
echo __CLASS__;
}
}
B::test();
?>
以上例程會(huì )輸出:
B
注意:
在非靜態(tài)環(huán)境下,所調用的類(lèi)即為該對象實(shí)例所屬的類(lèi)。由于
$this->
會(huì )在同一作用范圍內嘗試調用私有方法,而static::
則可能給出不同結果。另一個(gè)區別是static::
只能用于靜態(tài)屬性。
示例 #3 非靜態(tài)環(huán)境下使用 static::
<?php
class A {
private function foo() {
echo "success!\n";
}
public function test() {
$this->foo();
static::foo();
}
}
class B extends A {
/* foo() 將復制給 B,因此它的作用域將是 A 并調用成功 */
}
class C extends A {
private function foo() {
/* 替換原來(lái)的方法;新的作用域是 C */
}
}
$b = new B();
$b->test();
$c = new C();
$c->test(); //fails
?>
以上例程會(huì )輸出:
success! success! success! Fatal error: Call to private method C::foo() from context 'A' in /tmp/test.php on line 9
注意:
后期靜態(tài)綁定的解析會(huì )一直到取得一個(gè)完全解析了的靜態(tài)調用為止。另一方面,如果靜態(tài)調用使用
parent::
或者self::
將轉發(fā)調用信息。示例 #4 轉發(fā)和非轉發(fā)調用
<?php
class A {
public static function foo() {
static::who();
}
public static function who() {
echo __CLASS__."\n";
}
}
class B extends A {
public static function test() {
A::foo();
parent::foo();
self::foo();
}
public static function who() {
echo __CLASS__."\n";
}
}
class C extends B {
public static function who() {
echo __CLASS__."\n";
}
}
C::test();
?>以上例程會(huì )輸出:
A C C