array_reduce配合array_merge仅适用于结构规整的二维数组;混用索引与关联数组会导致键覆盖,应改用array_merge_recursive或foreach;含脏数据需先过滤再强转;性能上foreach更优。

用 array_reduce 降维含子数组的二维数组
直接上结论:如果子数组结构规整(全是数值或全是关联键),array_reduce 配合 array_merge 是最简洁的方案;但一旦混用索引数组和关联数组,array_merge 会覆盖键名,导致数据丢失——这是最容易踩的坑。
典型错误写法:
array_reduce($arr, function($carry, $item) {
return array_merge($carry, $item);
}, []);
问题在于:array_merge 对数字键会重排,对字符串键会覆盖。比如 ['a' => 1] 和 ['a' => 2] 合并后只剩 ['a' => 2]。
- 若所有子数组都是纯数值索引,可用
array_merge(...$arr)(PHP 5.6+),更高效且语义清晰 - 若含关联键且需保留全部元素(不覆盖),必须改用
array_merge_recursive或手动遍历foreach -
array_reduce的初始值必须是[],传null或省略会导致第一个子数组被跳过
处理含空子数组或非数组元素的健壮写法
真实数据常含 null、string、空数组等脏数据,直接 array_merge 会报 Warning: array_merge(): Argument #2 is not an array。
立即学习“PHP免费学习笔记(深入)”;
安全做法是先过滤再合并:
$filtered = array_filter($arr, 'is_array');
$result = array_reduce($filtered, function($carry, $item) {
return array_merge($carry, (array)$item);
}, []);
-
is_array过滤掉非数组项,避免警告 -
(array)$item强转确保每个子项可被array_merge处理(如null变成空数组) - 注意:强转对对象默认只取 public 属性,且不递归,慎用于复杂对象
需要保留原始键名时别用 array_merge
当子数组用的是有意义的字符串键(如 ['id' => 1, 'name' => 'A']),且你希望结果里每个元素仍保持独立结构(不是扁平化成一维键值对),那“降维”目标其实是「展开」而非「合并」——此时 array_merge 完全错误。
正确做法是用 array_merge(...$arr)(仅限 PHP 5.6+)或循环 push:
$result = [];
foreach ($arr as $sub) {
if (is_array($sub)) {
foreach ($sub as $item) {
$result[] = $item;
}
}
}
- 上面循环写法明确控制追加逻辑,不依赖键名,兼容所有 PHP 版本
-
array_merge(...$arr)在 PHP 7.4+ 中支持 unpacking,性能优于array_reduce - 若子数组本身是关联结构(如
['user' => [...], 'profile' => [...]]),那根本不是二维数组,而是嵌套结构——该用递归,不是降维
性能差异:小数组无所谓,大数组慎用 array_reduce
array_reduce 每次回调都新建数组并复制内容,时间复杂度 O(n²);而 foreach + array_push 是 O(n),实测万级元素时慢 3–5 倍。
- 1000 个子数组,每个 10 个元素:两者差异不明显
- 10000 个子数组:
foreach稳定快,array_reduce内存占用高且易触发 GC - 如果后续还要对结果做
array_unique或array_search,建议一开始就用foreach控制结构,避免中间数组反复创建
var_dump 看两眼结构再说。











