当咱们使用foreach时,内部究竟发生了什么(PHP5)?

如下全部结论均基于PHP5版本
看下面一段最基础的foreach遍历数组代码。php

<?php 
$arr = array(‘a’,’b’,’c’);
foreach ($arr as $key=> $value) {
    echo $key,$value,’<br/>’; //output :  0a1b2c
}
?>

输出为’0a1b2c’天然没有疑问,那么此过程当中$arr,$key,$value到底是通过怎样的运算,才输出这个结果的呢?数组

其实foreach遍历过程当中,并非直接操做$arr(原数组)的,而是会将$arr复制出一个$arrcopy(是一个$arr的一个复制品,我这里以$arrcopy代替),foreach在遍历过程当中操做的其实一直是$arrcopy。
注:关于$arrcopy这个值咱们是没办法提取出来的,由于这是我给他的命名,并无存在这个变量,可是foreach遍历过程当中确实会产生这么一个副本,这儿为了方便讲述我用$arrCopy表明。
Foreach遍历大概的流程是这样(伪代码):函数

<?php 
//伪代码
$arr = array('a','b','c');
/* foreach循环开始*/
//first loop
$arrCopy = $arr; //复制出一个待循环数组的副本,接下来都是操做这个副本
$key = currentKey($arrCopy); //将获取到的值分配给$k;
$val = currentVal($arrCopy); //将获取到的值分配给$v;
next($arrCopy);//移动副本数组的指针
$arr = $arrCopy;//将副本赋值回给$arr((主要是将指针同步移动))
//大括号内容
{
    echo $key,$value,’<br/>’;
}
//firt loop end

//second loop 
$key = currentKey($arrCopy); //将获取到的值分配给$k;
$val = currentVal($arrCopy); //将获取到的值分配给$v;
………
//seconde loop end
?>

这就是foreach代码的运行流程,总结一句话就是foreach遍历操做的时候并非原始数组,而是一个拷贝数组,可是每次循环的结尾都会将副本从新赋值回给原数组$arr = $arrCopy;。oop

如何证实个人说法呢?能够用下面这段代码检验。spa

<?php 
// $a = array('a','b','c');
 $arr = array('a','b','c');
foreach ($arr as $key=> $value) {
    $arr[] = 'd';
    print_r($arr);
    var_dump($key,$value);
}
?>

输出结果为:指针

//output:
Array
    (
    [0] => a
    [1] => b
    [2] => c
    [3] => d
    )
    int(0)
    string(1) "a"
    Array
    (
    [0] => a
    [1] => b
    [2] => c
    [3] => d
    [4] => d
    )
    int(1)
    string(1) "b"
    Array
    (
    [0] => a
    [1] => b
    [2] => c
    [3] => d
    [4] => d
    [5] => d
    )
    int(2)
    string(1) "c"

同窗们看出来了吗?
$arr数组的键值对一直在在增长,但是$key,$value的值到了int(2),string(1) “c”就结束了,并无如咱们所料的将值为d的那些键值对打印出来。
这儿就能证实,foreach遍历过程操做的是$arr的副本($arrcopy)。code

对了,foreach使用过程当中还有一些小地方须要注意。
例如foreach遍历数组的指针问题:blog

<?php 
$arr = array('a','b','c');
var_dump(current($arr)); //output:string(1) "a"
foreach ($arr as $key=> $value) {

}
var_dump(current($arr)); //output:bool(false)
?>

两次输出,不同的结果。为何呢?由于foreach循环遍历后的数组,该数组的指针是指向末尾的(此处的话指针就是在’c’的右边),而且使用完毕后不会帮咱们复位,因此咱们var_dump(current($arr))为 bool(false)。
那么在这里咱们须要特别注意,为了保险起见咱们在foreach遍历数组后,最好手动reset()一下数组,防止出错:图片

<?php 
$arr = array('a','b','c');
var_dump(current($arr)); //output:string(1) "a"
foreach ($arr as $key=> $value) {

}
reset($arr);
var_dump(current($arr)); / output:string(1) "a"
?>

这样就正常了。同步

还有一点PHP手册也提醒咱们了:
图片描述
转成代码的意思就是:

<?php 
$arr = array('a','b','c');
foreach ($arr as $key=> $value) {

}
var_dump($key);
var_dump($value);
?>

Foreach遍历后,$key和$value是真实存在的,最好使用后能手动unset()掉。

总结:foreach算是PHP里面比较复杂的一个函数了,由于牵扯到PHP底层的C语言的结构体,引用(is_ref__gc),指针移动……,因此在使用foreach的时候必定要特别注意啊!

相关文章
相关标签/搜索