array_filter()默认过滤所有falsy值(如0、"0"、null、""等),需用自定义回调明确排除null和空字符串,推荐写法:array_filter($arr, function($v) { return !is_null($v) && $v !== ''; })。

PHP中用array_filter()会误删0值
直接调用array_filter($arr)默认用布尔判断,0、"0"、0.0、空数组[]都会被当成false过滤掉——这不是你想要的“只去空值”,而是“去所有falsy值”。
常见错误现象:["a", 0, "b", null, ""] 经array_filter()后变成["a", "b"],0没了。
- 真正想保留的:数字
0、字符串"0"、0.0、false(如果业务逻辑需要) - 真正想剔除的:通常只是
null、""(空字符串)、[](空数组)——具体看需求,但绝不是所有falsy - 解决办法:传入自定义回调,显式比对要排除的值,例如
array_filter($arr, function($v) { return $v !== null && $v !== ''; })
保留0但去掉null和空字符串的稳妥写法
多数场景下,“空值”指null、""、有时还包括undefined(PHP里不存在,但可能来自JSON解码的null),而0、"0"、false需原样留下。
推荐写法(明确、无歧义):
立即学习“PHP免费学习笔记(深入)”;
$clean = array_filter($arr, function($v) {
return !is_null($v) && $v !== '';
});
- 不用
empty():它会把0、"0"、false、[]全判为真/假,不可控 - 不用
strlen($v) > 0:对null会触发warning,对数组会报错 - 如果还要排除空白字符串(如
" "),改用trim($v) !== '',但注意trim(null)返回"",所以仍要先判is_null
处理JSON来源数据时0变null的陷阱
前端传{"count": 0},PHP用json_decode($json, true)后$data['count']确实是0——但若前端传的是{"count": ""}或字段缺失,就可能是null。这时候“去空值”逻辑必须区分来源。
- 检查字段是否存在:
isset($data['count'])比!empty($data['count'])安全得多 - 若字段允许
0且必须保留,过滤前先做类型断言:is_int($data['count']) || is_string($data['count']) - 批量处理时,避免用
array_walk_recursive()套默认array_filter,容易层层误删
性能与可读性兼顾的封装建议
项目里反复出现这类逻辑,建议封装一个语义清晰的函数,而不是到处写匿名函数。
function filter_null_and_empty($arr) {
return array_filter($arr, function($v) {
return !is_null($v) && (is_string($v) ? trim($v) !== '' : true);
});
}
- 这个版本保留所有非字符串值(包括
0、false、[]),只对字符串做trim校验 - 如果连空数组也要剔除,加一句
!is_array($v) || !empty($v),但注意empty([])是true,所以得写!is_array($v) || count($v) > 0 - 别为了“简洁”用
array_diff($arr, [null, '']):它不递归,不处理嵌套,且0 == ''在松散比较下成立,有风险
最易被忽略的一点:过滤后的数组键可能不连续,如果后续要用数字索引遍历,记得补上array_values()。











