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 這兩個(gè)修飾的屬性的,而在class內(nèi)部是有權(quán)限訪問,foreach可以遍歷所有的屬性。
今天在寫PDO的時(shí)候發(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里面就一個(gè)public屬性 queryString 并且foreach也沒有出現(xiàn)這個(gè)值,這樣的情況就不是簡單的對屬性進(jìn)行遍歷了,而是class繼承了iterator迭代器,在foreach的時(shí)候會執(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 實(shí)現(xiàn)iterator的接口,foreach調(diào)用的時(shí)候會使用這個(gè)接口方法,調(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的處理過程是一種默認(rèn)方法,如果是繼承iterator的class被foreach遍歷的時(shí)候是上面這種方式
由此情況去套用 PDO的寫法還是行不通,因?yàn)槿绻覀僾ar_dump上面的哪個(gè)test類結(jié)果是這樣的:
test Object
(
[a] => a
[data:test:private] => Array
(
[0] => apple
[1] => banlance
[2] => current
)
[point:test:private] => 0
)
但是當(dāng)我們var_dump $db->query返回的對象時(shí)并沒有見到point這個(gè)iterator接口中定義的屬性以及遍歷的數(shù)據(jù) $data;
由此我們可以猜測PDOStatement繼承了一種迭代的接口但是并不是iterator
查看手冊可以發(fā)現(xiàn):
PDOStatement implements Traversable
查看Traversable的介紹如下圖:
![](http://img.jbzj.com/file_images/article/202104/2021421101942713.jpg?2021321101951)
由此明白了,PDOStatement的迭代實(shí)現(xiàn)都是在內(nèi)部,繼承iterator是php腳本的實(shí)現(xiàn)方式。
大致總結(jié)下:
foreach是可以遍歷數(shù)組的,也可以遍歷對象。對象只能羅列出public的屬性,如果想要foreach羅列出保護(hù)的屬性可以讓class繼承iterator并實(shí)現(xiàn)其中的方法,這樣foreach遍歷一個(gè)class的時(shí)候是按照class內(nèi)部實(shí)現(xiàn)的iterator進(jìn)行處理的。
-------------------------------------------------------------
PDO的問題:
PDO::query() 返回的是對象 PDOStatement (繼承的Traversable這個(gè)空接口,必須由Iterator or iteratorAggregate 接口實(shí)現(xiàn))。
PDOStatement 實(shí)現(xiàn)了Iterator接口的方法,其實(shí)現(xiàn)方法中操作的就是非public修飾的屬性,這個(gè)屬性里面存儲的是查詢結(jié)果集。
至此,foreach($db->query('sql..') as $row) 的執(zhí)行過程明白了
以上就是php中foreach遍歷類對象的總結(jié)的詳細(xì)內(nèi)容,更多關(guān)于php中foreach遍歷類對象的資料請關(guān)注腳本之家其它相關(guān)文章!
您可能感興趣的文章:- PHP面向?qū)ο蟪绦蛟O(shè)計(jì)之對象的遍歷操作示例
- PHP簡單遍歷對象示例
- PHP var_dump遍歷對象屬性的函數(shù)與應(yīng)用代碼
- php遍歷對象的方法