PHP foreach原理详解
一、foreach简介
1.foreach的遍历顺序
如果是索引数组,你会发现遍历出来的顺序并不是按索引大小遍历,而是按添加的顺序
,如果按照索引大小遍历,应该使用for
,而不是foreach
$arr[2]='中'; $arr[1]='国'; foreach($arr as $value){ echo $value; } 结果:中国
所以foreach遍历数组的顺序是由元素的添加顺序决定的,不管是索引数组还是关联数组
2.
当 foreach 开始执行时,数组内部的指针会自动指向第一个单元
。这意味着不需要在 foreach 循环之前调用 reset()怎么来理解这个呢?
$arr = array(1,2,3); foreach($arr as $k=>$v){ } var_dump(current($arr)); foreach($arr as $key=>$value){ echo $value." "; } var_dump(current($arr)); 结果:boolean false 1 2 3 boolean false
第一个foreach已经把指针移到尾部去了,并且试图努力的往后移动指针,直到移出界(current($arr)返回false),foreach结束foreach结束后,并没有帮我们把指针初始化,不然current应该返回数组的第一个单元,第二个foreach并没有受第一个foreach的影响,当foreach开始执行时,数组内部的指针会自动指向第一个单元。
$key = currentKey($arrCopy); //将获取到的值分配给$k; $val = currentVal($arrCopy); //将获取到的值分配给$v; next($arrCopy);//移动副本数组的指针 $arr = $arrCopy;//将副本赋值回给$arr((主要是将指针同步移动))
技术细节:当本次赋值给key和val之后,按照流程指针已经向下移动了一位
,所以当执行var_dump(current($arr));时打印false。如果移动指针的结果超出了数组单元的末端,则 next() 返回 FALSE。但foreach循环的次数不是副本数组的长度
二、加深foreach理解
$arr = array('a'=>1,'b'=>2,'c'=>3); foreach($arr as $k=>$v){ $v*=2; echo $v."<br />"; } var_dump($arr); foreach($arr as $key=>$value){ $arr[$key]=$value*2; } var_dump($arr); //传入& foreach($arr as &$v){ $v=$v*2; }
var_dump($arr);
结果:
2 4 6 array (size=3) 'a' => int 1 'b' => int 2 'c' => int 3 array (size=3) 'a' => int 2 'b' => int 4 'c' => int 6 array (size=3) 'a' => int 4 'b' => int 8 'c' => &int 12
原因分析:$k
和$v
都是临时变量,foreach的时候,把每个数组单元的键分别赋值给$k,把每个数组单元的值分别赋给$v,相等于$v=$arr[$k],$v*2仅仅是改变了$v的值(非&传递),并不会影响到$arr[$k],自然也就不会影响到$arr
而用第二种方法(引用)的时候,相等于$v=&$arr[$k],$arr[$k]和$v指向同一内存地址
,$v*2自然就改变了$arr[$k]的值,也就改变了$arr的值
PHP current() 函数语法
参考文档:深入探讨php的foreach
参考文档:当我们使用foreach时,内部究竟发生了什么(PHP5)?