用 array_reduce 合并二维数组需手动构造键名以保留原始结构,避免 array_merge 导致的键丢失或覆盖;foreach 更安全可控,尤其处理非标准结构或需过滤、动态命名时。
array_reduce 合并二维数组并保留原始键名直接用 array_reduce 做二维转一维时,默认会重置键(变成数字索引),想保留原 $arr[$outer_key][$inner_key] 的结构,必须手动拼接键名或嵌套构造。核心是:不依赖 array_merge(它会丢键),而是用赋值方式累积。
常见错误是写成:array_reduce($arr, 'array_merge', []) —— 这会导致所有子数组的键被扁平覆盖,且外层数字键丢失。
$outer_key . '_' . $inner_key 或类似规则生成新键user_0_name),在回调里做字符串拼接[$outer_key => $inner_value] 形式逐个合并foreach 手动遍历最可控,尤其处理非标准二维结构当二维数组不是严格「每项都是关联数组」(比如含空数组、数值索引混用、深层嵌套),array_reduce 容易报 Warning: array_merge(): Argument #2 is not an array。这时 foreach 显式判断更安全。
示例场景:从 API 返回的 ['data' => [['id'=>1,'name'=>'a'], ['id'=>2]]] 中提取所有 id 并带来源索引:
$result = [];
foreach ($data['data'] as $i => $item) {
if (isset($item['id'])) {
$result["row_{$i}"] = $item['id'];
}
}
array_reduce 回调中频繁类型判断带来的可读性下降var_dump 中间状态array_walk_recursive?不行,它会丢掉父级键信息array_walk_recursive 确实能把多维数组“走到底”,但它只提供当前值和叶子键(leaf key),完全无法获知这个值原本在第几层、属于哪个外层 key。所以它不满足「保留键值」的需求。
比如: ['user'=>['profile'=>['name'=>'Alice']] 经 array_walk_recursive 后只能得到 ['name'=>'Alice,
user 和 profile 全部丢失。
foreach 更复杂id)还会互相覆盖不管用哪种方法,只要把不同子数组的相同键(比如都叫 name)扁平到同一层,就必然面临覆盖问题。PHP 数组赋值是后写覆盖前写,所以顺序决定最终结果。
$arr['a']['id'] = 1 和 $arr['b']['id'] = 2,直接合并后只剩一个 id
a_id, b_id)或改用嵌套结构(['a'=>['id'=>1], 'b'=>['id'=>2]])