
is_int() 不能直接判断数组键是不是数字
PHP 数组的键可以是整数或字符串,但 is_int() 只能判断变量本身的类型,而数组键在遍历时会被自动转换为整数或字符串——哪怕你写的是 $arr[123],遍历中拿到的键可能是 int(123),也可能是 string("123"),取决于它最初怎么被插入的。所以直接对键调用 is_int($key) 看似合理,实则不可靠。
常见错误现象:
– 用 foreach ($arr as $key => $val) 遍历后对 $key 调 is_int(),结果部分数字键返回 false
– 从 JSON 解码来的数组(如 json_decode($json, true)),所有键都是 string 类型,哪怕看起来像数字
- 真正要判断的是“这个键是否**语义上代表一个整数索引**”,而不是“当前变量类型是不是 int”
- 更稳妥的方式是:先确认键是字符串,再用
ctype_digit()或正则判断是否纯数字;如果是 int,则天然符合 - 注意负数和带前导零的数字(如
"007"):它们不是有效数组索引,ctype_digit("007")返回true,但 PHP 数组不认为这是“数字键”(实际存储为字符串键)
array_keys() + filter_var() 是最稳的数字键筛选方式
如果你需要从一个混合键数组里挑出所有“真正数字键”(即 PHP 内部视为 integer key 的那些),不能依赖遍历时的 $key 类型,而应该回到数组结构本身——用 array_keys() 拿到所有键,再逐个验证。
使用场景:
– 清洗用户传入的关联/索引混用数组
– 实现类似 array_values() 但只保留数字键对应值的逻辑
– 序列化前校验键合法性
立即学习“PHP免费学习笔记(深入)”;
-
filter_var($key, FILTER_VALIDATE_INT)能同时处理 int 和 string 类型的键,且默认拒绝前导零、空格、负数(除非显式加FILTER_FLAG_ALLOW_NEGATIVE) - 比
is_numeric()更安全:后者会把"1e2"、"0x1F"甚至" 42 "当作数字,但这些都不是合法数组数字键 - 示例:
array_filter(array_keys($arr), function($k) { return filter_var($k, FILTER_VALIDATE_INT) !== false; });
isset($arr[$int_key]) 不等于 $int_key 是数字键
有人误以为只要 isset($arr[123]) 返回 true,就说明 123 是这个数组的“数字键”。其实不然——PHP 允许用整数去访问字符串键,只要该字符串能无损转成整数(比如 $arr["123"] = 'ok',那么 isset($arr[123]) 也会返回 true)。
这会导致两个问题:
– 错把字符串键当数字键用(比如后续做 array_merge() 时行为异常)
– 在反射或调试时看到“键存在”,却无法用 array_keys() 拿到它(因为实际键是 "123",不是 123)
- 验证一个整数是否是“原生数字键”,唯一可靠方式是:先取全部键
array_keys($arr),再检查该整数是否在其中且类型为 int - 简单判断可用:
in_array($int_key, array_keys($arr), true)
—— 注意必须用严格比较(第三个参数true),否则in_array(123, ["123"], true)返回false - 性能提示:对大数组频繁这么查不划算,建议一次提取所有数字键缓存复用
JSON 解码后所有键都是字符串,别信 is_int()
这是最容易踩的坑。PHP 的 json_decode($json, true) 总是把对象属性名转成字符串键,哪怕原始 JSON 是 {"0":"a","1":"b"},解码后键也是 "0"、"1",类型为 string。
后果:
– is_int(key) 全部失败
– array_values() 会重排,但你原本想保留“索引语义”就丢了
– 传给某些只认数字键的函数(如 implode() 的某些封装)可能出错
- 修复方法:手动映射一遍,用
array_walk_key()(PHP 8.4+)或循环重建:$fixed = []; foreach ($arr as $k => $v) { if (filter_var($k, FILTER_VALIDATE_INT) !== false) $fixed[(int)$k] = $v; else $fixed[$k] = $v; } - 如果确定输入 JSON 的键全是连续非负整数,更简单:用
array_values()重置,但会丢弃原始键名 - 兼容性提醒:PHP 8.1+ 支持
JSON_INVALID_UTF8_IGNORE,但不影响键类型;键类型问题跟编码无关,只跟 JSON 规范有关(JSON 标准里对象键只能是字符串)
数字键的“数字性”不在值,而在声明方式和内部存储类型。PHP 不提供运行时反射键类型的 API,所以所有判断都得绕道数据特征或重建逻辑。这点很容易被忽略,尤其在处理外部输入时。











