php中foreach遍歷類物件的總結
阿新 • • 發佈:2019-01-31
foreach 遍歷陣列很常見,同樣foreach也可以遍歷物件
做如下測試:
class my { public $a = 'a'; protected $b = 'b'; private $c = 'c'; private $data = array('fantasy','windows','linux'); // 內部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>'; // 內部foreach遍歷class $m->traversable();
輸出結果如下:
a=>a
--------------------------
a=>a
b=>b
c=>c
data=>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>';
}
輸出結果如下:
apple
banlance
current
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) 的執行過程明白了