foreach遍历不到新增元素因其基于数组快照;引用遍历时需foreach($arr as &$v)并unset($v)才能修改原数组;性能差异小,可读性优先;对象需实现Traversable接口才支持foreach。

foreach 为什么遍历不到新增的数组元素
因为 foreach 遍历的是数组的**快照**,不是实时引用。循环开始时 PHP 就把当前键值对拷贝进内部指针,后续在循环体里用 array_push() 或直接赋值往原数组加元素,不会影响本轮迭代。
- 常见错误现象:
foreach ($arr as $k => $v) { $arr[] = 'new'; }—— 新增项不会被这次循环处理到 - 如果真要边遍历边修改,改用
for+count(),但注意可能死循环(比如每次循环都加一个) - 更安全的做法是先收集要追加的数据,循环结束后再合并:
$to_add = []; foreach (...) { $to_add[] = ...; } $arr = array_merge($arr, $to_add);
引用遍历(&$v)改值后原数组没变?
只写 &$v 不够,$v 是值副本的引用,但若原数组是索引数组且你用的是 foreach ($arr as $v) 形式,PHP 会自动“脱引”——也就是实际操作的是临时变量,原数组不受影响。
- 必须同时满足两个条件才真正修改原数组:
foreach ($arr as &$v)(带 &)+ 循环结束后加unset($v) - 不加
unset($v)的后果:下一次循环或后续代码中,$v仍指向最后一个数组元素,导致意外覆盖,比如:$arr = [1,2,3]; foreach ($arr as &$v) {} $v = 99; // 结果 $arr 变成 [1,2,99] - 关联数组同理,但要注意键名不变,
foreach ($arr as $k => &$v)才能通过$v改对应值
foreach 和 for 性能差多少?
在绝大多数业务场景下,差异可以忽略。但底层机制不同: foreach 直接走哈希表迭代器,for 要反复查键、取值、判断边界。
- 小数组(
- 大数组(>10万)+ 频繁调用:实测
foreach快 10%~30%,尤其当数组稀疏或有空洞时,for的$i++可能跳过不存在的键 - 别为了“理论上更快”强行改写 —— 可读性和维护性更重要;只有压测发现它真是瓶颈,再考虑
foreach 遇到对象或 Iterator 怎么办
PHP 的 foreach 本质是调用对象的 getIterator() 方法,所以只要对象实现了 Iterator 或 Traversable 接口,就能用 foreach 遍历。
立即学习“PHP免费学习笔记(深入)”;
- 常见错误:直接
foreach ($obj as $k => $v)报错Fatal error: Uncaught Error: Cannot use object of type Xxx as array—— 说明该对象没实现遍历接口 - 修复方式:在类里加
implements Iterator并实现 5 个方法,或者更简单地用ArrayObject包一层:new ArrayObject($obj->toArray()) - 注意:
foreach对对象默认只遍历 public 属性,private/protected 不出现,和var_dump()行为一致
unset($v),还有以为边循环边 $arr[] = 能立刻生效。这两个点没意识到,调试半小时都找不到根因。











