foreach 遍曆數組很常見,同樣foreach也可以遍曆對象
做如下測試:
class my{public $a = 'a';protected $b = 'b';private $c = 'c';private $data = array('fantasy','windows','linux');// 內部foreach遍曆classfunction traversable(){foreach($this as $key=>$val){echo $key.'=>';print_r($val);echo '<br>';}}}$m = new my();// 外部foreach遍曆classforeach($m as $key=>$val){echo $key.'=>';print_r($val);echo '<br>';}echo '--------------------------<br>';// 內部foreach遍曆class$m->traversable();
輸出結果如下:
a=>a--------------------------a=>ab=>bc=>cdata=>Array ( [0] => fantasy [1] => windows [2] => linux )
由此可知,對於外部的foreach遍曆是沒有許可權訪問 protected private 這兩個修飾的屬性的,而在class內部是有許可權訪問,foreach可以遍曆所有的屬性。
今天在寫PDO的時候發現可以這樣寫:
foreach($db->query('SELECT * FROM tab') as $row){ print_r($row);}
這樣快速的擷取了全部查詢結果,可奇怪的是$this->query() 返回的是 object類型 PDOStatement ,var_dump()列印出來的結果是這樣的:
object(PDOStatement)#2 (1) { ["queryString"]=> string(18) "SELECT * FROM user"}
PDOStatement裡面就一個public屬性 queryString 並且foreach也沒有出現這個值,這樣的情況就不是簡單的對屬性進行遍曆了,而是class繼承了iterator迭代器,在foreach的時候會執行class裡面的迭代方式,遍曆迭代器指定的資料
關於迭代器看下面的例子:
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>';}
輸出結果如下:
applebanlancecurrent
test class 實現iterator的介面,foreach調用的時候會使用這個介面方法,調用過程大致如下面虛擬碼:
// 迭代過程虛擬碼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類結果是這樣的:
test Object( [a] => a [data:test:private] => Array ( [0] => apple [1] => banlance [2] => current ) [point:test:private] => 0)
但是當我們var_dump $db->query返回的對象時並沒有見到point這個iterator介面中定義的屬性以及遍曆的資料 $data;
由此我們可以猜測PDOStatement繼承了一種迭代的介面但是並不是iterator
查看手冊可以發現:
PDOStatement implements Traversable
查看Traversable的介紹如下圖:
由此明白了,PDOStatement的迭代實現都是在內部,繼承iterator是php指令碼的實現方式。
大致總結下:
foreach是可以遍曆數組的,也可以遍曆對象。對象只能羅列出public的屬性,如果想要foreach羅列出保護的屬性可以讓class繼承iterator並實現其中的方法,這樣foreach遍曆一個class的時候是按照class內部實現的iterator進行處理的。
-------------------------------------------------------------
PDO的問題:
PDO::query() 返回的是對象 PDOStatement (繼承的Traversable這個空介面,必須由Iterator or iteratorAggregate 介面實現)。
PDOStatement 實現了Iterator介面的方法,其實現方法中操作的就是非public修飾的屬性,這個屬性裡面儲存的是查詢結果集。
至此,foreach($db->query('sql..') as $row) 的執行過程明白了