PHP中each与数组变量分离的那些事情

看有人遇到问题,尝试寻找了下问题的原因。


在写一个方法的时候,遇到这个问题,将问题简化下,如下:

[php]  view plain copy
  1. $arr = array(1,2,3,4,5,6);  
  2. $n = 3;  
  3.   
  4. foreach($arr as $key=>$val){  
  5.     if($val == $nbreak;  
  6. }  
  7. while(list($k,$v) = each($arr)){  
  8.     echo $v;  
  9. }  
由于 each() 循环不会自动重置的数组指针,那么上边代码执行输出:  456

但是写在函数中结果却不同了,如下:
[php]  view plain copy
  1. $arr = array(1,2,3,4,5,6);  
  2. $n = 3;  
  3. test1($arr,$n);  
  4. function test1($arr,$n) {  
  5.   
  6.   
  7.     foreach($arr as $key=>$val){  
  8.         if($val == $nbreak;  
  9.     }  
  10.     while(list($k,$v) = each($arr)){  
  11.         echo $v;  
  12.     }  
  13. }  
代码执行结果为:  123456


其实都是php的变量"写时复制(copy on write)"惹的祸,为什么这么说,且看分析:

什么是"写时复制(copy on write)"?可以参看PHP内核探索:变量的引用与计数规则 与深入理解PHP原理之变量分离/引用(Variables Separation),这里用一个例子说明:

[php]  view plain copy
  1. $arr = array(1,2,3,4,5,6);  
  2. $n = 3;  
  3.   
  4. $arr1 = $arr// 此时并没有复制,是引用关系  
  5. debug_zval_dump($arr); // recount = 3,因为debug_zval_dump有一次  
  6. $arr[0] = 1; // 此时有写操作,要分离  
  7. debug_zval_dump($arr); // recount = 2  

php手册上说:
因为将一个数组赋值给另一个数组时会重置原来的数组指针,因此在上边的例子中如果我们在循环内部将$fruit 赋给了另一个变量的话将会导致无限循环。
php手册关于each 介绍有提到,其实就是说数组变量分离的时候会重置数组的指针,看下面的例子:

[php]  view plain copy
  1. $arr = array(1,2,3,4,5,6);  
  2. $n = 3;  
  3.   
  4. $arr1 = $arr// 此时并没有复制,是引用关系,可以看作是函数复制调用  
  5.   
  6. foreach($arr as $key=>$val){  
  7.     if($val == $nbreak;  
  8. }  
  9.   
  10. debug_zval_dump($arr); // refcount = 3  
  11. list($k,$v) = each($arr); // 因为each会改变数组的指针,所以还是写操作,存在分离  
  12. echo $v;  
  13. debug_zval_dump($arr); // refcount = 2  


至于&操作的话,不存在“变量分离”,或者说“变量分离”是在&操作的同时完成的。

[php]  view plain copy
  1. $arr = array(1,2,3,4,5,6);  
  2. $n = 3;  
  3.   
  4. $arr1 = &$arr// 引用,“变量分离”是在&操作的同时完成  
  5.   
  6. foreach($arr as $key=>$val){  
  7.     if($val == $nbreak;  
  8. }  
  9.   
  10. debug_zval_dump($arr); // refcount = 2  
  11. list($k,$v) = each($arr); // 不存在变量分离  
  12. echo $v;  
  13. debug_zval_dump($arr); // refcount = 2  

相同的效果还有:

[php]  view plain copy
  1. $arr = array(1,2,3,4,5,6);  
  2. $n = 3;  
  3.   
  4. $arr1 = $arr; <span style="font-family: Arial, Helvetica, sans-serif;">// 此时并没有复制,是引用关系,可以看作是函数复制调用</span>  
  5.   
  6. foreach($arr as $key=> &$val){ // 存在可能写的情况,变量分离  
  7.     if($val == $nbreak;  
  8. }  
  9.   
  10. debug_zval_dump($arr); // refcount = 2  
  11. list($k,$v) = each($arr); // 不存在变量分离  
  12. echo $v;  
  13. debug_zval_dump($arr); // refcount = 2  
所以在使用each之前,最好是先用reset(如果存在引用会产生变量分离)统一重置数组指针,就可以避免上述问题。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值