foreach 遍历数组很常见,同样foreach也可以遍历对象
做如下测试:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
|
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
由此可知,对于外部的foreach遍历是没有权限访问 protected private 这两个修饰的属性的,而在class内部是有权限访问,foreach可以遍历所有的属性。
今天在写PDO的时候发现可以这样写:
1
2
3
4
|
foreach ( $db ->query( 'SELECT * FROM tab' ) as $row ) { print_r( $row ); } |
这样快速的获取了全部查询结果,可奇怪的是$this->query() 返回的是 object类型 PDOStatement ,var_dump()打印出来的结果是这样的:
1
2
3
4
|
object(PDOStatement)#2 (1) { [ "queryString" ]=> string(18) "SELECT * FROM user" } |
PDOStatement里面就一个public属性 queryString 并且foreach也没有出现这个值,这样的情况就不是简单的对属性进行遍历了,而是class继承了iterator迭代器,在foreach的时候会执行class里面的迭代方式,遍历迭代器指定的数据
关于迭代器看下面的例子:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
|
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
test class 实现iterator的接口,foreach调用的时候会使用这个接口方法,调用过程大致如下面伪代码:
1
2
3
4
5
6
7
|
// 迭代过程伪代码 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类结果是这样的:
1
2
3
4
5
6
7
8
9
10
11
12
|
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) 的执行过程明白了
以上就是php中foreach遍历类对象的总结的详细内容,更多关于php中foreach遍历类对象的资料请关注服务器之家其它相关文章!
原文链接:https://blog.csdn.net/fanyilong_v5/article/details/39008829