后期靜態(tài)綁定

PHP 增加了一個叫做后期靜態(tài)綁定的功能,用于在繼承范圍內引用靜態(tài)調用的類。

準確說,后期靜態(tài)綁定工作原理是存儲了在上一個“非轉發(fā)調用”(non-forwarding call)的類名。當進行靜態(tài)方法調用時,該類名即為明確指定的那個(通常在 :: 運算符左側部分);當進行非靜態(tài)方法調用時,即為該對象所屬的類。所謂的“轉發(fā)調用”(forwarding call)指的是通過以下幾種方式進行的靜態(tài)調用:self::,parent::,static:: 以及 forward_static_call()??捎?get_called_class() 函數來得到被調用的方法所在的類名,static:: 則指出了其范圍。

該功能從語言內部角度考慮被命名為“后期靜態(tài)綁定”?!昂笃诮壎ā钡囊馑际钦f,static:: 不再被解析為定義當前方法所在的類,而是在實際運行時計算的。也可以稱之為“靜態(tài)綁定”,因為它可以用于(但不限于)靜態(tài)方法的調用。

self:: 的限制

使用 self:: 或者 __CLASS__ 對當前類的靜態(tài)引用,取決于定義當前方法所在的類:

示例 #1 self:: 用法

<?php
class {
    public static function 
who() {
        echo 
__CLASS__;
    }
    public static function 
test() {
        
self::who();
    }
}

class 
extends {
    public static function 
who() {
        echo 
__CLASS__;
    }
}

B::test();
?>

以上例程會輸出:

A

后期靜態(tài)綁定的用法

后期靜態(tài)綁定本想通過引入一個新的關鍵字表示運行時最初調用的類來繞過限制。簡單地說,這個關鍵字能在上述例子中調用 test() 時引用的類是 B 而不是 A。最終決定不引入新的關鍵字,而是使用已經預留的 static 關鍵字。

示例 #2 static:: 簡單用法

<?php
class {
    public static function 
who() {
        echo 
__CLASS__;
    }
    public static function 
test() {
        static::
who(); // 后期靜態(tài)綁定從這里開始
    
}
}

class 
extends {
    public static function 
who() {
        echo 
__CLASS__;
    }
}

B::test();
?>

以上例程會輸出:

B

注意:

在非靜態(tài)環(huán)境下,所調用的類即為該對象實例所屬的類。由于 $this-> 會在同一作用范圍內嘗試調用私有方法,而 static:: 則可能給出不同結果。另一個區(qū)別是 static:: 只能用于靜態(tài)屬性。

示例 #3 非靜態(tài)環(huán)境下使用 static::

<?php
class {
    private function 
foo() {
        echo 
"success!\n";
    }
    public function 
test() {
        
$this->foo();
        static::
foo();
    }
}

class 
extends {
   
/* foo() 將復制給 B,因此它的作用域將是 A 并調用成功 */
}

class 
extends {
    private function 
foo() {
        
/* 替換原來的方法;新的作用域是 C */
    
}
}

$b = new B();
$b->test();
$c = new C();
$c->test();   //fails
?>

以上例程會輸出:

success!
success!
success!


Fatal error:  Call to private method C::foo() from context 'A' in /tmp/test.php on line 9

注意:

后期靜態(tài)綁定的解析會一直到取得一個完全解析了的靜態(tài)調用為止。另一方面,如果靜態(tài)調用使用 parent:: 或者 self:: 將轉發(fā)調用信息。

示例 #4 轉發(fā)和非轉發(fā)調用

<?php
class {
    public static function 
foo() {
        static::
who();
    }

    public static function 
who() {
        echo 
__CLASS__."\n";
    }
}

class 
extends {
    public static function 
test() {
        
A::foo();
        
parent::foo();
        
self::foo();
    }

    public static function 
who() {
        echo 
__CLASS__."\n";
    }
}
class 
extends {
    public static function 
who() {
        echo 
__CLASS__."\n";
    }
}

C::test();
?>

以上例程會輸出:

A
C
C