php二维转一维包含子数组_php含子数组降维array_reduce【教程】

6次阅读

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

php二维转一维包含子数组_php含子数组降维array_reduce【教程】

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 或省略会导致第一个子数组被跳过

处理含空子数组或非数组元素的健壮写法

真实数据常含 nullString、空数组等脏数据,直接 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_uniquearray_search,建议一开始就用 foreach 控制结构,避免中间数组反复创建

实际用哪一种,取决于你的子数组是否真“二维”、键名是否重要、数据是否干净——别被“一行解决”的诱惑带偏,先 var_dump 看两眼结构再说。

text=ZqhQzanResources