foreach
循环中回调函数的累积问题及解决方案在PHP开发中,foreach
循环结合回调函数时,如果回调函数使用了静态变量,可能会导致结果在每次循环迭代中累积,产生错误的结果。本文将分析此问题并提供解决方案。
以下代码片段展示了这个问题:foreach
循环调用getChildAreaId
回调函数,但由于getChildAreaId
函数内部使用了静态变量$res
,导致结果累积。
foreach ($towns as $key => $val) { $areaidarr = getChildAreaId($val['id']); $result[$val['name']] = $worker::where('area_id', 'in', $areaidarr)->count(); } function getChildAreaId($id) { static $area; $area = $area ?? new appcommonmodelarea; $result = collection($area->where(['pid' => $id])->order('id desc')->select())->toArray(); static $res = []; if ($result) { foreach ($result as $key => $val) { $res[] = $val['id']; getChildAreaId($val['id']); // 递归调用 } } return $res; }
每次循环调用getChildAreaId
,$res
都会累加之前的结果,最终导致$result
包含错误的数据。
问题的根源在于getChildAreaId
函数中使用了静态变量$res
。静态变量在函数调用之间保持其值,因此每次递归调用都会修改同一个$res
数组,导致结果累积。
解决方法主要有两种:
消除静态变量: 重写getChildAreaId
函数,避免使用静态变量$res
。 可以使用局部变量,并在递归调用时正确合并结果。
重置静态变量: 在每次foreach
循环迭代之前,手动重置getChildAreaId
函数中的静态变量$res
。但这是一种不太优雅的解决方案,容易出错。
推荐使用第一种方法,重写getChildAreaId
函数:
function getChildAreaId($id) { $area = new appcommonmodelarea; $result = collection($area->where(['pid' => $id])->order('id desc')->select())->toArray(); $res = []; if ($result) { foreach ($result as $key => $val) { $res[] = $val['id']; $res = array_merge($res, getChildAreaId($val['id'])); } } return $res; }
在这个修改后的版本中,$res
变成了局部变量,每次函数调用都会创建一个新的空数组,避免了结果累积。 递归调用时,使用array_merge
函数正确合并子结果。 这保证了每次foreach
循环都得到独立且正确的结果。
通过以上修改,可以有效解决foreach
循环中回调函数导致结果累积的问题,确保程序的正确性和可靠性。