php 8.0+ 推荐用 array_filter() 配合 str_starts_with() 按键名前缀过滤,需传 array_filter_use_key;低版本用 strpos() 或 substr() 手动判断,注意键名大小写、空格及标志位遗漏问题。

直接用 array_filter() 配合 str_starts_with() 最干净,PHP 8.0+ 推荐;低版本得用 strpos() 或 substr() 手动判断。
用 array_filter() + str_starts_with()(PHP 8.0+)
这是最直观的方式:遍历键名,保留前缀匹配的项。注意必须用 array_keys($arr, null, true) 或直接传回调访问键名,因为默认 array_filter() 只传值。
- 必须启用键名传递:回调函数第二个参数是键名,需在
array_filter()第三个参数传ARRAY_FILTER_USE_KEY -
str_starts_with($key, 'prefix_')比substr($key, 0, 7) === 'prefix_'更安全,不担心空键或长度不足 - 不会改变原数组顺序,键名保留原样
php
$arr = ['prefix_a' => 1, 'other' => 2, 'prefix_b' => 3];
$filtered = array_filter($arr, function($v, $k) {
return str_starts_with($k, 'prefix_');
}, ARRAY_FILTER_USE_KEY);
// 结果:['prefix_a' => 1, 'prefix_b' => 3]
兼容 PHP strpos())
strpos($key, 'prefix_') === 0 是最常用替代方案,但要注意它返回 false 和 0 都是“falsy”,所以必须用全等判断。
- 不能写成
!strpos($key, 'prefix_')—— 键名恰好以prefix_开头时返回0,会被误判为 false - 如果前缀为空字符串,
strpos返回0,逻辑仍成立;但通常前缀非空,不必额外防御 - 比
preg_match('/^prefix_/', $key)快得多,无正则开销
php
$filtered = array_filter($arr, function($v, $k) {
return strpos($k, 'prefix_') === 0;
}, ARRAY_FILTER_USE_KEY);
想批量提取多个前缀?别硬写多重 array_filter()
如果要按不同前缀分组(比如同时筛出 user_*、post_*、meta_*),用一次遍历 + foreach 更高效,避免反复扫描数组。
立即学习“PHP免费学习笔记(深入)”;
-
array_filter()每调用一次就完整遍历一遍,N 个前缀就是 N 次 O(n) 操作 - 单次
foreach+ 多个if分支,时间复杂度稳定 O(n) - 结果可存在多维数组里:
$groups['user_'][] = [$k => $v],或直接构建新键名映射
注意键名大小写和空白字符
PHP 数组键区分大小写,'Prefix_a' 不会被 str_starts_with($k, 'prefix_') 匹配。如果业务允许模糊匹配,得先统一处理:
- 转小写再比:
str_starts_with(strtolower($k), 'prefix_') - 但要注意:原始键名会丢失大小写,若后续还要用原键,就得提前保存
- 键名开头/结尾含空格?PHP 一般不允许(解析时就被截断或报错),但动态生成键时可能混入
\t或\0,建议用trim($k)防御
真正容易漏的是 ARRAY_FILTER_USE_KEY 这个标志位——忘了加它,回调根本收不到键名,所有筛选都变成按值操作,结果完全不对。











