array_filter() 默认保留逻辑为真的值(如1、"hello"、[1,2]),过滤假值(false、null、0、0.0、""、"0"、[]);需严格筛选true时必须传回调函数。

用 array_filter() 筛选布尔值为真的元素
array_filter() 默认行为就是保留“逻辑上为真”的值,但要注意:它不是判断 === true,而是做松散的布尔转换(即 if ($value) 的结果)。这意味着 1、"hello"、[1,2] 会被留下,而 0、""、null、[](空数组)会被过滤掉。
实操建议:
- 如果只要严格等于
true的项(比如数组里混有true、false、1、"true"),必须传入回调函数:array_filter($arr, function($v) { return $v === true; }) - 默认不传回调时,
0.0和-0也会被当成假值剔除,这和某些语言不同,需留意 - 键名默认保留,如需重排索引,包一层
array_values()
区分 true 和“逻辑真”是关键陷阱
常见错误是以为 array_filter($arr) 只留 true,结果发现 42、"0"(注意:字符串 "0" 是假值!)、[null] 全都混进来了。PHP 的“假值”列表固定且明确:false、null、0、0.0、""、"0"、[] —— 其余均为真。
使用场景举例:
立即学习“PHP免费学习笔记(深入)”;
- 处理表单 checkbox 多选结果:
$_POST['tags']可能是[0 => 'php', 1 => 'js'],此时用默认array_filter()没问题;但若字段本身存的是布尔标记(如['active' => true, 'deleted' => false]),就必须用严格比较 - API 返回的混合数据中提取“启用状态为 true”的条目,别漏掉
"1"这类字符串型真值是否要纳入
性能与兼容性提醒
array_filter() 在 PHP 5.6+ 中对大数组性能尚可,但若数组超 10 万项且只想要严格 true,自己写 foreach 循环并用 === true 判断,反而更直观、更容易控制内存(避免生成中间数组)。
注意点:
- PHP 8.0+ 支持
ARRAY_FILTER_USE_BOTH标志,可用于同时检查键和值,但筛选布尔真通常不需要 - 不要在回调里修改原数组,
array_filter()不会传引用,改了也白改 - 如果源数组含对象,对象恒为真(除非实现
__isset或__toString并返回假值),这点容易误判
为什么 "0" 被当假值却常被忽略
这是最隐蔽的坑:字符串 "0" 在 PHP 布尔上下文中是 false,但直觉上它是“非空字符串”,很多人会漏掉这个特例。调试时若发现某个本该出现的 ID(比如数据库里存的字符串 "0" 表示未分配)莫名其妙消失,大概率就是这里卡住了。
验证方式很简单:
var_dump((bool)"0"); // bool(false)
应对方法:
- 若业务中
"0"是合法值,就别依赖默认array_filter(),改用显式回调:function($v) { return $v !== "" && $v !== null && $v !== false; }(按需调整) - 入库前统一类型(如全转成 int 或全用
null表示缺失),比过滤时补救更可靠
布尔真筛选看着简单,实际得先想清楚:你要的是语言层面的“真值”,还是业务定义的“有效标记”。后者往往需要脱离默认行为单独处理。











