本篇文章的內容是PHP後期綁定,現在分享給大家,有需要的朋友也可以參考一下本篇的文章了
php 後期靜態繫結
自 PHP 5.3.0 起,PHP 增加了一個叫做後期靜態繫結的功能,用於在繼承範圍內引
用靜態調用的類。
準確說,後期靜態繫結工作原理是儲存了在上一個“非轉寄調用”(non-forwarding
call)的類名。當進行靜態方法調用時,該類名即為明確指定的那個(通常在 ::
運算子左側部分);當進行非靜態方法調用時,即為該對象所屬的類。所謂的“轉寄調用”(forwarding
call)指的是通過以下幾種方式進行的靜態調用:self::,parent::,static:: 以及
forward_static_call()。可用 get_called_class() 函數來得到被調用的方法所在的類名,static::
則指出了其範圍。
該功能從語言內部角度考慮被命名為“後期靜態繫結”。“後期綁定”的意思是說,static::
不再被解析為定義當前方法所在的類,而是在實際運行時計算的。也可以稱之為“靜態繫結”,因為它可以用於(但不限於)靜態方法的調用。
self
使用 self:: 或者 CLASS 對當前類的靜態引用,取決於定義當前方法所在的類:
Example #1 self:: 用法
<?phpclass 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(); //輸出A?>
static後期靜態繫結的用法
後期靜態繫結本想通過引入一個新的關鍵字表示運行時最初調用的類來繞過限制。簡單地說,這個關鍵字能夠讓你在上述例子中調用 test() 時引用的類是 B 而不是 A。最終決定不引入新的關鍵字,而是使用已經預留的 static 關鍵字。
<?phpclass A { public static function who() { echo __CLASS__; } public static function test() { static::who(); // 後期靜態繫結從這裡開始 }}class B extends A { public static function who() { echo __CLASS__; }}B::test(); //輸出B?>
Note:
在非靜態環境下,所調用的類即為該對象執行個體所屬的類。由於 $this-> 會在同一作用範圍內嘗試調用私人方法,而 static::
則可能給出不同結果。另一個區別是 static:: 只能用於靜態屬性。
非靜態環境下使用 static::
<?phpclass A { private function foo() { echo "success!\n"; } public function test() { $this->foo(); static::foo(); }}class B extends A { /* foo() will be copied to B, hence its scope will still be A and * the call be successful */}class C extends A { private function foo() { /* original method is replaced; the scope of the new one is 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
Note:
後期靜態繫結的解析會一直到取得一個完全解析了的靜態調用為止。另一方面,如果靜態調用使用 parent:: 或者 self:: 將轉寄調用資訊。
轉寄和非轉寄調用
<?phpclass 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();?>
以上常式會輸出:
A C C