最稳妥方案是用 array_replace_recursive() 填充空值,它递归替换 null、''、[] 而保留 0/false 等合法值;需预处理统一空值类型,配合自定义 is_blank() 判断逻辑。

PHP中用 array_replace_recursive() 填充空值最稳妥
直接用 array_merge() 会丢键、覆盖非空值,而 array_replace_recursive() 能递归穿透多维数组,只替换 null、''、[] 这类“空”值,保留已有有效数据。注意它不处理 0、false、0.0 —— 这是合理设计,不是 bug。
实操建议:
- 先定义默认值数组,结构与目标数组一致,所有字段填期望的 fallback 值
- 对原数组做预处理:用
array_map()+ 匿名函数把null和空字符串转成NULL(统一类型),避免类型松散比较出错 - 调用
array_replace_recursive($defaults, $data),顺序不能反,否则默认值被覆盖
判断“空”的逻辑必须自己控制,PHP 没有内置统一标准
empty() 和 !$val 会把 0、"0"、false 都当空,但业务中这些常是合法值。比如商品库存为 0 不能替换成 999。
推荐写个辅助函数明确语义:
立即学习“PHP免费学习笔记(深入)”;
function is_blank($val): bool {
return $val === null || $val === '' || (is_array($val) && count($val) === 0);
}
再配合 array_walk_recursive() 手动遍历替换:
- 适合需要精细控制的场景,比如只替换字符串型空值,跳过数字型
0 - 注意
array_walk_recursive()不支持引用修改多维数组的键,得用引用传参或改用foreach+ 递归函数 - 性能比
array_replace_recursive()略低,但可控性高
用 ?? 和 ?: 只适用于单层变量,别硬套到数组遍历里
看到有人写 $arr['name'] ?: '未知' 就以为能推广到整个数组,结果写出 array_map(fn($v) => $v ?: 'default', $arr) —— 这会把 0、false 全干掉。
更糟的是嵌套时写 $arr['user']['age'] ?? 18,一旦 $arr['user'] 是 null,PHP 7.4+ 会报 Trying to access array offset on value of type null。
安全写法只有两种:
- 用
isset($arr['user']['age']) ? $arr['user']['age'] : 18 - 升级到 PHP 8.0+,用空合并链操作符:
$arr['user']['age'] ?? $arr['user']['default_age'] ?? 18
JSON 场景下要注意 null 和缺失字段的区别
前端传来的 JSON 如果字段没填,PHP 解析后可能是 key 不存在,也可能是 key => null。这两种情况在填充时要分开对待。
比如 json_decode('{"name":"Tom"}', true) 没有 email 键;而 json_decode('{"name":"Tom","email":null}', true) 有 email 键且值为 null。
所以填充前先用 array_key_exists() 判断键是否存在,再结合 is_null() 判断值是否为空,不能只依赖 isset() —— 它对 null 值返回 false。
实际项目里,经常要先跑一遍 array_fill_keys($required_keys, null) 把缺的键补上,再统一处理空值。










