關(guān)于 PHP 的 new static 延遲靜態(tài)綁定,或者叫后期靜態(tài)綁定,在 Laravel 中遇到一個(gè)使用上的問(wèn)題。如下,在 Laravel 中調(diào)用 Model 新增數(shù)據(jù)的時(shí)候,首先給 Model 加了一個(gè)獲取分表的方法:
protected function addToMessage($msgType, $userID, $commentID, $replyCommentID, $replyUserID, $gameID) { if (!$userID) { return false; } $table = 't_message_' . hashID($userID, 100); $this->message->setTable($table)->create([ 'msg_type' => $msgType, 'user_id' => $userID, 'comment_id' => $commentID, 'reply_comment_id' => $replyCommentID, 'reply_user_id' => $replyUserID, 'game_id' => $gameID, 'is_read' => 0, 'created_at' => date('Y-m-d H:i:s'), ]); return true; }
這里 setTable 方法是在 Model 里定義的獲取分表的方法:
public function setTable($table) { $this->table = $table; return $this; }
從報(bào)錯(cuò)日志中發(fā)現(xiàn) $this->table 并沒(méi)有生效,但實(shí)際上在調(diào)用 create 方法之前打印表名的時(shí)候是期望的值,這里調(diào)用 create 方法為什么 $this->table 沒(méi)有被重置呢?
這里 $this->message 是一個(gè)繼承 Model 類(lèi)的模型類(lèi),其中 create 方法:
public static function create(array $attributes = []) { $model = new static($attributes); $model->save(); return $model; }
位于 vendor\laravel\framework\src\Illuminate\Database\Eloquent\Model.php Line 557.
因?yàn)?Laravel 框架的這個(gè) Model 類(lèi)是一個(gè) abstract 類(lèi)型,PHP 中 abstract 類(lèi)可以用 new static 后期靜態(tài)綁定的方式實(shí)例化,而 create 方法里 $model = new static($attributes) 實(shí)際上就是重新實(shí)例化了并返回,而調(diào)用者 Model 類(lèi)沒(méi)有定義 table 屬性,所以這個(gè)時(shí)候 $this->table 是沒(méi)有值的。
解決辦法是用 save 方法即可,如圖所示。實(shí)際上 create 方法也調(diào)用了 save 方法。
實(shí)驗(yàn)
一個(gè)抽象類(lèi) A,有個(gè) create 方法,通過(guò)延遲靜態(tài)綁定實(shí)例化并返回。B 類(lèi)繼承 A,test 方法中修改父類(lèi)的 name 屬性。
<?php abstract class A { protected $name = "tanteng"; public static function create() { return new static(); } } class B extends A { //protected $name = '紙牌屋弗蘭克'; public function test() { $this->name = "Tony Tan"; return $this; } } $obj1 = (new B)->test(); $obj2 = (new B)->test()->create(); var_dump($obj1); var_dump($obj2);
結(jié)果顯示 $obj1 和 $obj2 這兩個(gè)實(shí)例都是 B 的實(shí)例,調(diào)用 test 方法屬性 name 改變了,但是調(diào)用 create 方法后,name 屬性并沒(méi)有改變。 這也就是在本文中說(shuō)的在 Lavarel 中遇到的場(chǎng)景。 (這里如果把注釋打開(kāi),打印的 name 就是重寫(xiě)的值)
如果把抽象類(lèi) A 改成普通類(lèi),new static 改成 new self 的方式實(shí)例化,結(jié)果就不同了,打印的屬性 name 都是各自類(lèi)的屬性。
參考鏈接
PHP:后期靜態(tài)綁定
PHP中new self()和new static()的區(qū)別
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持。