array_merge_recursive 不能按需合并相同键的值,必须手动遍历处理;推荐用 foreach + isset 实现数值累加、字符串拼接或数组收集,并注意键类型、空值和类型安全校验。

PHP 二维数组按键合并成一维:用 array_merge_recursive 还是手动遍历?
直接说结论:array_merge_recursive 不能真正“合并相同键的值”,它只是把同名键的值塞进一个数组里,遇到数字键还会重排索引。如果你要的是「相同字符串键的值相加/拼接/去重」,必须手动处理。
常见错误现象:array_merge_recursive(['a' => 1], ['a' => 2]) 返回 ['a' => [1, 2]],不是 ['a' => 3] 或 ['a' => '1,2'] —— 这往往不是你想要的结果。
- 适用场景:原始数据是多个关联数组(如数据库查出的多条记录),需按字段名归并统计或拼接
- 关键点:先明确“相同键”怎么合并——求和?字符串连接?取最大值?保留第一个?
- 性能影响:纯 PHP 循环比内置函数慢一点,但可控;用
foreach+isset比array_key_exists更快
手动合并字符串键:用 foreach + isset 累加或拼接
这是最灵活、最常用的方式。假设你要把二维数组中所有同名键的数值相加:
$data = [
['price' => 100, 'tax' => 10],
['price' => 200, 'tax' => 20],
['price' => 150, 'discount' => 5]
];
$result = [];
foreach ($data as $row) {
foreach ($row as $key => $value) {
if (isset($result[$key])) {
$result[$key] += $value; // 数值累加
} else {
$result[$key] = $value;
}
}
}
// 得到 ['price' => 450, 'tax' => 30, 'discount' => 5]
- 如果要字符串拼接,把
$result[$key] += $value换成$result[$key] = ($result[$key] ?? '') . $value - 如果要保留数组(如收集所有
id),用$result[$key][] = $value - 注意:
isset($result[$key])比array_key_exists快,且不会触发未定义索引警告
避免踩坑:数字键、空值、类型混用怎么办?
二维数组里若含数字键(如 [0 => 'a', 1 => 'b']),直接用上面逻辑会覆盖——因为 $result[0] 被反复赋值。你需要提前过滤或区分键类型。
立即学习“PHP免费学习笔记(深入)”;
- 只处理字符串键:在内层循环加判断
is_string($key) - 空值处理:用
$value !== null && $value !== ''控制是否参与合并 - 类型安全:数值累加前用
is_numeric($value)校验,否则'1' + '2'是 3,但'abc' + 1会变 1(静默转整型) - 键名大小写敏感:PHP 数组键严格区分大小写,
'Price'和'price'是两个键
用 array_reduce 写法更函数式,但可读性未必更好
如果你偏好函数式风格,可以用 array_reduce 替代外层 foreach:
$result = array_reduce($data, function ($carry, $item) {
foreach ($item as $k => $v) {
if (is_string($k) && !isset($carry[$k])) {
$carry[$k] = $v;
} elseif (is_string($k) && is_numeric($v) && is_numeric($carry[$k])) {
$carry[$k] += $v;
}
}
return $carry;
}, []);
这种写法省了一行初始化,但调试时不如传统循环直观;嵌套深了容易漏掉边界条件,比如没处理 $carry[$k] 已存在但 $v 不是数字的情况。
真正复杂的需求(如多级键映射、条件跳过、动态合并策略)建议封装成函数,把合并逻辑抽出来,而不是堆砌在循环里。键名冲突和类型隐式转换,才是最容易被忽略、又最难 debug 的地方。











