php面向對象編程筆記static的後期靜態綁定

最近在看laravel的源碼,開始的時候static關鍵字的用法讓我讀完一頭霧水,

static不應該是用來定義靜態變量方法和屬性的麼?


在方法中

static::setInstance($this);

是什麼?


查了手冊才明白,這是static關鍵字的另一種用法,“後期靜態綁定”,也可以稱爲“靜態綁定”

http://php.net/manual/zh/language.oop5.late-static-bindings.php


官方的介紹:

該功能從語言內部角度考慮被命名爲“後期靜態綁定”。“後期綁定”的意思是說,static::  不再被解析爲定義當前方法所在的類,而是在實際運行時計算的。也可以稱之爲“靜態綁定”,因爲它可以用於(但不限於)靜態方法的調用。  


官方的幾個例子很好的說明了後期靜態綁定的意思


self::的限制

使用 self:: 或者 __CLASS__ 對當前類的靜態引用,取決於定義當前方法所在的類:

Example #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();
?>

輸出:A


後期靜態綁定的用法

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

Example #2 static:: 簡單用法

<?php
class 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


在這裏static相當於self,parent一樣的關鍵字,用來調用方法,不過與self不同,static調用的類並不一定是靜態屬性和靜態方法,這點上看更接近於parent這個關鍵字的用法。


but,在下面的例子了還看到這樣一種用法

<?php
class Model
{
    public static function find()
    {
        echo static::$name;
    }
}
class Product extends Model
{
    protected static $name = 'Product';
}
Product::find();
?>

在父類裏並未定義$name這個屬性,這時候在子類裏必須

static $name

不過一般程序應該不會出現這種父類沒定義屬性而直接使用的情況吧。

-----------------------------------------------------------------------------------------------------------


另外,測試例子的時候想到這麼一個情況

class A {
    public static function who() {
        echo __CLASS__;
    }
    public function test() {
        static::who();
    }
}
class B extends A {
    public static function who() {
        echo __CLASS__;
    }
}
(new B())->test();

非靜態方法可以調用靜態方法,反之則不行


laravel設計模式中大量使用了這種靜態綁定的方法,尤其是核心功能Facades。

php面向對象編程還有很多東西需要了解。。。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章