本篇文章給大家介紹一下在PHP中檢測一個類是否可以被foreach遍歷的方法。有一定的參考價值,有需要的朋友可以參考一下,希望對大家有所幫助。
在PHP中檢測一個類是否可以被foreach遍歷
在PHP中,我們可以非常簡單的判斷一個變量是什么類型,也可以非常方便的確定一個數組的長度從而決定這個數組是否可以遍歷。那么類呢?我們要如何知道這個類是否可以通過 foreach 來進行遍歷呢?其實,PHP已經為我們提供了一個現成的接口。
class Obj1 { public $v = 'V:Obj1'; private $prv = 'prv:Obj1'; } $obj1 = new Obj1(); echo $obj1 instanceof Traversable ? 'yes' : 'no', PHP_EOL; // no class Obj2 implements IteratorAggregate { public $v = 'V:Obj2'; private $prv = 'prv:Obj2'; public function getIterator() { return new ArrayIterator([ 'v' => $this->v, 'prv' => $this->prv, ]); } } $obj2 = new Obj2(); echo $obj2 instanceof Traversable ? 'yes' : 'no', PHP_EOL; // yes
從上面的例子中可以看出,每一個 $obj1 無法通過 Traversable 判斷,所以它是不能被遍歷的。而第二個 $obj2 則是實現了迭代器接口,這個對象是可以通過 Traversable 判斷的。在PHP手冊中,Traversable 接口正是用于檢測一個類是否可以被 foreach 遍歷的接口。
這個接口有幾個特點:
-
實現此接口的內建類可以使用 foreach 進行遍歷而無需實現 IteratorAggregate 或 Iterator 接口。
-
這是一個無法在 PHP 腳本中實現的內部引擎接口。IteratorAggregate 或 Iterator 接口可以用來代替它。
也就是說這個接口不需要我們去手工實現,只需要我們的類實現迭代器相關的接口就可以通過這個接口的驗證的判斷。如果單獨去實現這個接口的話,將會報錯并提示我們應該去實現 IteratorAggregate 或 Iterator 接口。
// Fatal error: Class ImplTraversable must implement interface Traversable as part of either Iterator or IteratorAggregate in Unknown class ImplTraversable implements Traversable{ }
其實之前的文章中,我們已經驗證過,對象是可以被遍歷的,而且并不需要實現什么迭代器接口就可以被 foreach 遍歷。它會輸出 所有 public 的屬性。
// foreach foreach ($obj1 as $o1) { echo $o1, PHP_EOL; } foreach ($obj2 as $o2) { echo $o2, PHP_EOL; } // V:Obj1 // V:Obj2 // prv:Obj2
也就是說這個 Traversable 接口的作用在實際使用中并不明顯。相信我們決大部分人也并沒有使用過這個接口來判斷過類是否可以被遍歷。但是從上面的例子中我們可以看出,迭代器能夠自定義我們需要輸出的內容。相對來說比直接的對象遍歷更加的靈活可控。另外,如果是數組強轉對象的情況,Traversable 接口同樣無法進行判斷。
$arr = [1, 2, 3, 4]; $obj3 = (object) $arr; echo $obj3 instanceof Traversable ? 'yes' : 'no', PHP_EOL; // no foreach ($obj3 as $o3) { echo $o3, PHP_EOL; }
其實,數組本身就是天然的可迭代對象。這里雖然進行了類型強轉,但其實應該將數組強轉的對象視為默認實現了迭代的器的對象更合適。當然,這類接口更大的意義還是在于代碼規范及強制檢查方面。
測試代碼:
https://github.com/zhangyue0503/dev-blog/blob/master/php/202003/source/%E5%9C%A8PHP%E4%B8%AD%E6%A3%80%E6%B5%8B%E4%B8%80%E4%B8%AA%E7%B1%BB%E6%98%AF%E5%90%A6%E5%8F%AF%E4%BB%A5%E8%A2%ABforeach%E9%81%8D%E5%8E%86.php
推薦學習:php視頻教程