如果你是一個懶惰的程式員,你看到以下代碼可能會惱火 abstract class U{ }class u1 extends U{ public static function create(){ return new u1(); } } class u2 extends U{ public static function create(){ return new u2(); }} 這段代碼正常工作是沒問題,但大量重複的代碼會很煩人 我不想在每個子類中添加create方法,如果把create方法放在超類U中,代碼可能是 abstract class U{ public static function create(){ return new self(); }}class u1 extends U{ function a(){}} class u2 extends U{ }u1::create(); 看起來很優雅整潔,現在我們把常見代碼放在一個位置,並用self作為對該類的引用。但這裡我們對self做了一個假設。 實際上,self對該類所起的作用與$this對對象所起的作用並不完全相同。self指的不是調用上下文,他指的是解析上下文,因此如果運行上面的列子,將會得到 Fatal error: Cannot instantiate abstract class U in D:\wamp\www\test\oop\static.php on line 21 因此self被解析為定義create的U,而不是解析為調用self的u1類。 php5.3之前,在這方面都有嚴格的限制,產生過很多笨拙的解決方案,php5.3引入了延遲靜態繫結 及使用 關鍵字 static static類似self,但它指的是被調用的類而不是包含類。 在以下例子中u1::create將產生u1對象,而不是執行個體化U對象 abstract class U{ public static function create(){ return new static(); }}class u1 extends U{} class u2 extends U{ }u1::create(); static不僅可以用於執行個體化,和self,parent一樣還可以作為靜態方法的調用標識符,甚至是從非靜態上下文中調用 abstract class U{ private $group; public function __construct(){ $this->group=static::getGroup(); } public static function create(){ return new static(); } static function getGroup(){ return 'default'; }}class u1 extends U{} class u2 extends U{ static function getGroup(){ return 'u2'; }}class u3 extends u2{ }print_r(u1::create());echo '<br/>';print_r(u3::create()); u1 Object ( [group:U:private] => default ) u3 Object ( [group:U:private] => u2 )