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) 的执行过程明白了