最直接的方式是用 array_filter() 配合匿名函数筛选长度达标元素,需根据数据类型选择 strlen()、count() 或类型守卫,并注意多字节字符串应使用 mb_strlen()。

用 array_filter() 配合匿名函数筛选长度达标元素
PHP 中最直接的方式是用 array_filter() 对数组元素逐个判断长度。注意:这里“长度”需明确是字符串长度(strlen())、数组元素个数(count()),还是其他自定义逻辑——不同数据类型不能混用同一判断方式。
常见错误是直接对混合类型数组调用 strlen(),导致 Warning: strlen() expects parameter 1 to be string, array given。
- 若元素全是字符串,用
strlen($item) >= $minLen - 若元素全是子数组,用
count($item) >= $minLen - 若类型不确定,先用
is_string()或is_array()做类型守卫
$arr = ['a', 'hello', ['x', 'y'], 'hi'];
$filtered = array_filter($arr, function($item) {
return is_string($item) && strlen($item) >= 3;
});
// 结果:['hello']
用 array_walk() 或 foreach 手动构建新数组更可控
当需要同时处理长度判断 + 其他逻辑(如记录键名、跳过空值、兼容 PHP 5.6 等老版本),array_filter() 的隐式键保留可能反而是负担。此时显式遍历更稳妥。
关键点:不要在循环中用 unset() 原数组——会破坏键连续性且不易调试;应新建数组并按需 [] 追加。
立即学习“PHP免费学习笔记(深入)”;
-
foreach更易读,适合多数场景 -
array_walk()需传引用(&$result)才能写入外部变量,稍易出错 - 若需保持原始键名,用
$result[$key] = $item;若要重排索引,用$result[] = $item
$arr = ['ab' => 'hi', 'cd' => 'world', 'ef' => []];
$result = [];
foreach ($arr as $k => $v) {
if (is_string($v) && strlen($v) > 2) {
$result[$k] = $v; // 保留键名
}
}
// $result = ['cd' => 'world']
注意 mb_strlen() 和多字节字符的坑
如果数组里有中文、emoji 或其他 UTF-8 字符串,用 strlen() 会返回字节数而非字符数,导致长度判断失准。例如 strlen('你好') 返回 6(UTF-8 下每个汉字占 3 字节),但实际字符数是 2。
此时必须换用 mb_strlen($item, 'UTF-8'),并确保 mbstring 扩展已启用(否则报 Fatal error: Uncaught Error: Call to undefined function mb_strlen())。
- 检查扩展是否开启:
extension_loaded('mbstring') - 不指定编码时,
mb_strlen()可能依赖mb_internal_encoding(),建议显式传参 - 性能上
mb_strlen()比strlen()略慢,纯 ASCII 场景无需强切
避免用 array_map() 替代筛选逻辑
新手容易误用 array_map() 来“转换+过滤”,比如写成 array_map(function($x) { return strlen($x) > 3 ? $x : null; }, $arr)。这不会真正过滤掉元素,只是把不达标的变成 null,数组长度不变,还引入了脏值。
真正筛选必须用 array_filter() 或显式遍历。若想链式操作,可组合 array_values(array_filter(...)) 重置索引。
-
array_filter()默认会移除值为false、0、''、null的项——这是陷阱,务必传入完整回调函数 - PHP 7.4+ 支持箭头函数,但注意它不能访问
$this且不支持引用,简单长度判断够用
strlen() 必崩。先做 is_string() 守卫,再决定用 strlen() 还是 mb_strlen(),比事后 debug 快得多。











