PHP中使用foreach和引用导致程序BUG的问题介绍

foreach 循环实例用法,在PHP中的foreach是用来进行循环对一个数组的所有元素,作者的foreach基本语法如下:

  1. FOREACH ($array_variable as $value)
  2. {
  3. [code to execute]
  4. }
  5. or
  6. FOREACH ($array_variable as $key => $value)
  7. {
  8. [code to execute]
  9. }

在这两种情况下,多次[代码执行]将被处死是等于在$ array_variable数组元素的个数,让我们来看一个例子。假设我们有下面的代码段:

  1. $array1 = array(1,2,3,4,5);
  2. FOREACH ($array1 as $abc)
  3. {
  4. print "new value is " . $abc*10 . "<br>";
  5. }

输出结果

  1. new value is 10
  2. new value is 20
  3. new value is 30
  4. new value is 40
  5. new value is 50

上面是我们对foreach常常使用的方法,下面再看实例

  1. $a = array(1, 2);
  2. $b = array(11, 12);
  3. foreach($a as &$r){
  4. }
  5. foreach($b as $r){
  6. }
  7. echo $a[1]; // 输出 12

两个的循环的本意可能是: 第一个循环需要在循环中修改元素的内容, 所以使用引用; 但第二个循环只是把 $r 当作一个临时变量. 可是, 为什么 $a[1] 的值发生了改变呢?

当对 $a 的迭代完成后, $r 是 $a[1] 的引用, 改变 $r 的值, 就是改变 $a[1]. 这时, 你可以会奇怪, 代码中并没有修改 $r, 也没有修改 $a[1] 呀?

其实, foreach 是操作的是数组的拷贝, 所以, 后一个迭代相当于:

  1. for($i=0; $i<count($b); $i++){
  2. $r = $b[$i]; // 修改了 $r! 相当于 $a[1] = $b[$i];
  3. }

为了避免这种情况, 应该在第一个迭代之后, 执行 unset($r);从当前环境中删除 $r 这个变量(引用变量).即使不是前面的例子, 在第一个迭代之后, 仍然十分可能再执行类似的语句:$r = 123;

循环变量一般是临时变量, 同一个变量名在代码不同的地方表示不同的东西, 但变量的作用域又存在于循环之外. 这就是这种作用域规则的坏处, 加上”变量不声明即使用”的坏外, 再加上变量无类型的坏处.

所以, 在 PHP 中使用引用变量, 应该在引用使用完之后, 应该 unset(). 所有变量在使用之前应该先 unset().

总结:PHP 引用有些类似 C 语言指针, 但一些重要的特性和 C 语言指针不一样, 如果不注意, 会导致程序 BUG. foreach 操作的是数组或对象的拷贝, 但 PHP5, 可以使用引用操作对象元素本身