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

为什么 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() 合并,别混淆场景。











