先用 array_filter() 清理假值,再用 array_unique() 去重;因 array_filter() 默认过滤 null、false、0、''、0.0、[] 等 falsy 值,若需保留 0 或 '0' 等须传回调函数明确判断,且顺序不可颠倒。

直接结论:用 array_filter() 去空值,再套 array_unique() 去重,但顺序不能反,且需注意 array_filter() 的默认判定逻辑。
为什么不能先 array_unique() 再 array_filter()
因为 array_unique() 会保留第一个出现的元素,而空值(如 ''、0、false、null)如果排在前面,去重后仍会残留;后续 array_filter() 虽能清理,但若你依赖键名连续性或原始顺序,中间多一次去重反而打乱意图。更关键的是:没必要——空值本就不该参与“去重”的语义。
实操建议:
- 始终先
array_filter($arr)清理“假值”,再array_unique()处理剩余有效值 - 若需保留键名(比如关联数组),加
ARRAY_FILTER_USE_KEY或显式回调,避免误删合法0键 - 若数组含
0、'0'、false等需区分的值,绝不能依赖默认array_filter(),必须传回调函数明确判断
array_filter() 默认行为到底过滤哪些值
PHP 默认把以下值判为“空”并过滤掉:null、false、0、0.0、''(空字符串)、[](空数组)。这不是“去空字符串”而是“去所有 falsy 值”。
立即学习“PHP免费学习笔记(深入)”;
常见错误现象:
- 数组里有
['a', 0, 'b'],结果变成['a', 'b']—— 你本想留0却丢了 - 接口返回
['id' => 0, 'name' => 'test'],array_filter()后整个键值对消失
正确做法(只去空字符串和 null):
$clean = array_filter($arr, function($v) {
return $v !== '' && $v !== null;
});
array_unique() 的 SORT_REGULAR 和 SORT_STRING 怎么选
默认用 SORT_REGULAR,按 PHP 类型规则比较(0 == '0' 为 true),所以 [0, '0'] 去重只剩一个;而 SORT_STRING 强制转字符串比('0' === '0'),但 0 和 '0' 在字符串下也相等,依然去重。
真正要保留数字 0 和字符串 '0',得用严格类型区分:
- 先用
array_map('strval', $arr)统一转字符串再array_unique()—— 但丢失原始类型 - 或写自定义去重:遍历 +
spl_object_hash()(对象)或json_encode()(简单值)做键判重 - 更实际的做法:接受 PHP 的松散比较特性,提前规范输入类型,避免混存
0和'0'
一行写法与生产环境避坑点
看似简洁的一行:array_unique(array_filter($arr)),在生产中容易出问题:
- 没处理键名重排:
array_values()必须显式加,否则后续for循环可能越界 - 没考虑大数组性能:
array_filter()+array_unique()是两次全量遍历,万级数据无压力,但十万级建议改用foreach单次扫描 +isset($seen[$v])缓存判重 - JSON 场景特别注意:
json_decode($json, true)返回的数组若含null字段,array_filter()会直接删掉该键,前端可能报undefined
最稳的通用写法(兼顾可读与健壮):
$filtered = array_filter($arr, function($v) {
return is_string($v) ? $v !== '' : !is_null($v) && $v !== false;
});
$unique = array_values(array_unique($filtered));
复杂点永远在“空值”的定义上——是空字符串?是 null?是空白字符?还是业务意义上的无效值?别让函数替你猜。











