php怎样用递归二维转一维_php递归遍历二维降维代码【技巧】

9次阅读

Array_merge(…$arr)不能直接降维,因为它要求所有解包元素均为数组,而混合类型(如数字6)会触发警告;正确做法是用递归函数逐层判断并合并。

php怎样用递归二维转一维_php递归遍历二维降维代码【技巧】

为什么 array_merge(...$arr) 不能直接降维

很多人第一反应是用 array_merge 解包,但这是错的——它只对「一层数组元素」有效。如果 $arr[ [1,2], [3,[4,5]], 6 ]array_merge(...$arr) 会报 Warning: array_merge(): Argument #3 is not an array,因为 6 不是数组。递归不是为了炫技,而是必须处理「任意深度嵌套 + 混合类型(数字/字符串/数组)」的真实数据。

flatten_recursive() 的安全写法

核心是判断每个值是否为数组,是则递归,否则追加。关键点在于:不依赖 is_array() 的宽松判断(比如 is_array(NULL) 返回 false,但 is_array(0) 也返回 false,没问题),但要避免对对象、资源等非数组类型调用 foreach

  • is_iterable($item)is_array($item) 更稳妥(php 7.1+),能兼容 Traversable 对象
  • 若需兼容老版本 PHP,坚持用 is_array($item),但必须先 !is_string($item) 排除字符串(防止把 "123" 当数组遍历)
  • 避免在递归函数里用 &$result 引用传参,容易在多层调用中混淆作用域;推荐返回新数组

示例:

function flatten_recursive($arr) {     $result = [];     foreach ($arr as $item) {         if (is_array($item)) {             $result = array_merge($result, flatten_recursive($item));         } else {             $result[] = $item;         }     }     return $result; }

遇到 Maximum function nesting level 怎么办

这不是代码写错了,是 Xdebug 默认限制了递归深度(通常 256 层)。当处理深层嵌套(如解析某些 xml 转数组结果、无限级分类树)时容易触发。不要急着改代码逻辑,先确认是不是真需要那么深:

立即学习PHP免费学习笔记(深入)”;

  • count($arr, COUNT_RECURSIVE) 粗略估算总元素数,再看嵌套层数是否合理
  • 临时提高限制:在脚本开头加 ini_set('xdebug.max_nesting_level', 500);(仅开发环境
  • 生产环境建议改用模拟递归(while + array_pop),避免爆,尤其处理用户可控输入时

字符串键会被丢弃,这正常吗

完全正常。flatten_recursive() 的目标是「降维」,不是「保结构」。原始二维数组中的 ['name' => 'Tom', 'scores' => [90, 85]] 经过递归展开后变成 [ 'Tom', 90, 85 ],所有字符串键都丢失。如果你需要保留路径信息(比如转成 ['name' => 'Tom', 'scores.0' => 90]),那就不是降维,而是「扁平化带路径键」,得换思路——用 array_walk_recursive() 配合引用回调,或者手动拼接键名。

真正容易被忽略的是:有些业务数据表面是二维,实际是「关联数组的关联数组」,比如配置项 ['db' => ['host' => '127.0.0.1', 'port' => 3306]]。这种不该用递归降维,该用 array_merge_recursive() 合并,别混淆场景。

text=ZqhQzanResources