公司升級 php7 後出現了一個問題
類似這樣 isset($post->user->name) 始終為 false
之前的 php 5.6 就很正常
laravel 版本是 5.1.35(很久沒升級了)
先看看 isset
isset 用來檢測變數是否設定
首先我們來看官方的一個例子
大致上是下面這個意思
<?php class Post { protected $attributes = ['content' => 'foobar']; public function __get($key) { if (isset($this->attributes[$key])) { return $this->attributes[$key]; } } } $post = new Post(); echo isset($post->content); // false
上面這個例子將永遠返回 false,因為 foo 並不是 Post 的屬性,而是 __get 取出來的
魔術方法 __isset
那麼怎麼解決上面那個問題呢?使用魔術方法
<?PHP class Post { protected $attributes = ['content' => 'foobar']; public function __get($key) { if (isset($this->attributes[$key])) { return $this->attributes[$key]; } } public function __isset($key) { if (isset($this->attributes[$key])) { return true; } return false; } } $post = new Post(); echo isset($post->content); //true
類似 Eloquent 的例子
看著 laravel 5.1.35 的程式碼,我們自己寫一個簡單的例子
先有一個 Model,簡單的實現。__get,__set,__isset
class Model { // 存放屬性 protected $attributes = []; // 存放關係 protected $relations = []; public function __get($key) { if( isset($this->attributes[$key]) ) { return $this->attributes[$key]; } // 找到關聯的物件,放在關係裡面 if (method_exists($this, $key)) { $relation = $this->$method(); return $this->relations[$method] = $relation; } } public function __set($k, $v) { $this->attributes[$k] = $v; } public function __isset($key) { if (isset($this->attributes[$key]) || isset($this->relations[$key])) { return true; } return false; } }
然後我們定義一個 Post Moel 和一個 User Moel
class Post extends Model { protected function user() { $user = new User(); $user->name = 'user name'; return $user; } } class User extends Model { }
好了來驗證一下 isset
$post = new Post(); echo 'isset 發帖使用者:'; echo isset($post->user) ? 'true' : 'false'; // false echo PHP_EOL; echo 'isset 發帖使用者的名字:'; echo isset($post->user->name) ? 'true' : 'false'; // false echo PHP_EOL; echo '發帖使用者的名字:'; echo $post->user->name; // user name echo PHP_EOL; echo '再次判斷 isset 發帖使用者的名字:'; echo isset($post->user->name) ? 'true' : 'false'; // true echo PHP_EOL;
答案
分析上面的結果,感覺像是 php 7 isset 方法對物件的判斷有了變化,如果先執行一次,$post->user->name,也就是將 user 放在 post 的 relations 中,這樣 isset ($post->user) 為 true,隨後 isset ($post->user->name) 才為 true。
最後在 Eloquent model 的 git log 中 找到了答案,
PHP 7 has fixed a bug with __isset which affects both the
native isset and empty methods. This causes specific issues
with checking isset or empty on relations in Eloquent. In
PHP 7 checking if a property exists on an unloaded relation,
for example isset($this->relation->id) is always
returning false because unlike PHP 5.6, PHP 7 is now
checking the offset of each attribute before chaining to
the next one. In PHP 5.6 it would eager load the relation
without checking the offset. This change brings back the
intended behavior of the core Eloquent model __isset method
for PHP 7 so it works like it did in PHP 5.6.
For reference, please check the following link,
specifically Nikita Popov's comment (core PHP dev) -
https://bugs.php.net/bug.php?id=69659
大致上是 php7 isset 判斷的時候,會依次判斷。php5.6 則會預載入關係。其實 laravel 也早在 5 月份就做了相關的處理,所以升級 laravel 後,自然也就沒有這個問題了。
以上就是PHP7中的isset的詳細內容,更多請關注TW511.COM其它相關文章!