這篇文章給大家分享的是PHP中foreach遍歷類對象的實現(xiàn),foreach 遍歷數(shù)組很常見,同樣foreach也可以遍歷對象,文中示例代碼介紹的非常詳細,感興趣的朋友接下來一起跟隨小編看看吧。
做如下測試:
class my { public $a = 'a'; protected $b = 'b'; private $c = 'c'; private $data = array('fantasy','windows','linux'); // 內(nèi)部foreach遍歷class function traversable() { foreach($this as $key=>$val) { echo $key.'=>'; print_r($val); echo '<br>'; } } } $m = new my(); // 外部foreach遍歷class foreach($m as $key=>$val) { echo $key.'=>'; print_r($val); echo '<br>'; } echo '--------------------------<br>'; // 內(nèi)部foreach遍歷class $m->traversable();
輸出結(jié)果如下:
a=>a
————————–
a=>a
b=>b
c=>c
由此可知,對于外部的foreach遍歷是沒有權(quán)限訪問 protected private 這兩個修飾的屬性的,而在class內(nèi)部是有權(quán)限訪問,foreach可以遍歷所有的屬性。
今天在寫PDO的時候發(fā)現(xiàn)可以這樣寫:
foreach($db->query('SELECT * FROM tab') as $row) { print_r($row); }
這樣快速的獲取了全部查詢結(jié)果,可奇怪的是$this->query() 返回的是 object類型 PDOStatement ,var_dump()打印出來的結(jié)果是這樣的:
object(PDOStatement)#2 (1) { ["queryString"]=> string(18) "SELECT * FROM user" }
PDOStatement里面就一個public屬性 queryString 并且foreach也沒有出現(xiàn)這個值,這樣的情況就不是簡單的對屬性進行遍歷了,而是class繼承了iterator迭代器,在foreach的時候會執(zhí)行class里面的迭代方式,遍歷迭代器指定的數(shù)據(jù)
關(guān)于迭代器看下面的例子:
class test implements Iterator { public $a = 'a'; private $data = array('apple','banlance','current'); private $point = 0; public function __construct() { $this->point = 0; } public function current() { return $this->data[$this->point]; } public function key() { return $this->point; } public function next() { ++$this->point; } public function rewind() { $this->point=0; } public function valid() { return isset($this->data[$this->point]); } } $t = new test(); foreach($t as $val) { print_r($val); echo '<br>'; }
輸出結(jié)果如下:
apple
banlance
test class 實現(xiàn)iterator的接口,foreach調(diào)用的時候會使用這個接口方法,調(diào)用過程大致如下面?zhèn)未a:
// 迭代過程偽代碼 while(valid) { <span style="white-space:pre"> </span>current/key <span style="white-space:pre"> </span>next } rewind
so,之前的foreach對class的處理過程是一種默認方法,如果是繼承iterator的class被foreach遍歷的時候是上面這種方式
由此情況去套用 PDO的寫法還是行不通,因為如果我們var_dump上面的哪個test類結(jié)果是這樣的:
test Object ( [a] => a [data:test:private] => Array ( [0] => apple [1] => banlance [2] => current ) [point:test:private] => 0 )
但是當我們var_dump $db->query返回的對象時并沒有見到point這個iterator接口中定義的屬性以及遍歷的數(shù)據(jù) $data;
由此我們可以猜測PDOStatement繼承了一種迭代的接口但是并不是iterator
查看手冊可以發(fā)現(xiàn):
PDOStatement implements Traversable
查看Traversable的介紹如下圖:

由此明白了,PDOStatement的迭代實現(xiàn)都是在內(nèi)部,繼承iterator是php腳本的實現(xiàn)方式。
大致總結(jié)下:
foreach是可以遍歷數(shù)組的,也可以遍歷對象。對象只能羅列出public的屬性,如果想要foreach羅列出保護的屬性可以讓class繼承iterator并實現(xiàn)其中的方法,這樣foreach遍歷一個class的時候是按照class內(nèi)部實現(xiàn)的iterator進行處理的。
————————————————————-
PDO的問題:
PDO::query() 返回的是對象 PDOStatement (繼承的Traversable這個空接口,必須由Iterator or iteratorAggregate 接口實現(xiàn))。
PDOStatement 實現(xiàn)了Iterator接口的方法,其實現(xiàn)方法中操作的就是非public修飾的屬性,這個屬性里面存儲的是查詢結(jié)果集。
至此,foreach($db->query(‘sql..’) as $row) 的執(zhí)行過程明白了