强制转整型用 (int) 最直接但不安全:它截断小数、忽略非数字前缀、空串和 null 均变 0,不四舍五入也不报错;intval() 默认行为相同,仅多支持进制转换;严格校验应使用 filter_var($var, filter_validate_int)。

强制转整型用 (int) 最直接,但不是万能的
PHP 里把变量转成整数,(int) 是最常用也最直观的方式。它会截断小数、忽略非数字前缀、把空字符串和 null 变成 0,但不会做四舍五入,也不会报错——这点容易让人误以为“转成功了”,其实可能丢了关键信息。
常见错误现象:(int)"123abc" 得到 123,看着像对的,但 (int)"abc123" 却是 0;(int)"" 和 (int)null 都是 0,无法区分原始值到底是空还是真为零。
- 字符串开头有数字时,只取连续数字部分(
(int)"42px"→42) - 字符串开头无数字时,一律转成
0((int)"px42"→0) -
false转成0,true转成1,注意别和业务逻辑里的“是否启用”混淆 - 浮点数超范围(如
9E100)会变成0或最大整数,不提示溢出
intval() 比 (int) 多一个进制参数,但默认行为一样
intval() 看起来更“正式”,但它在默认调用下(不传第二个参数)和 (int) 完全等价:都是松散解析、截断、静默失败。唯一实用差异是支持进制转换,比如处理十六进制字符串。
使用场景:你明确知道输入是带前缀的进制字符串,比如 "0xFF" 或 "1010"(二进制),且想按对应进制转整数。
立即学习“PHP免费学习笔记(深入)”;
-
intval("0xFF", 16)→255,而(int)"0xFF"→0(因为开头不是数字) -
intval("1010", 2)→10,(int)"1010"也能得到10,但那是当十进制处理的 - 进制参数必须是 2–36 之间的整数,否则返回
0,不报错也不警告 - 如果字符串含非法字符(如
"10G"在 base36 下合法,但在 base10 下就停在"10"),intval()同样只取前面合法部分
需要严格校验时,别用强制转换,改用 filter_var()
当你的输入来自表单、API 或配置文件,且要求“要么是合法整数,要么明确失败”,(int) 和 intval() 都不合适——它们太宽容,会掩盖格式错误。
这时候用 filter_var($var, FILTER_VALIDATE_INT),它不转类型,只验证并返回原值或 false。配合 filter_var($var, FILTER_SANITIZE_NUMBER_INT) 可先清理再判断,但注意后者会删掉所有非数字字符(包括负号和小数点),慎用。
-
filter_var("123", FILTER_VALIDATE_INT)→123 -
filter_var("12.3", FILTER_VALIDATE_INT)→false(不是截成12) -
filter_var("-42", FILTER_VALIDATE_INT)→-42(支持负数) - 可加选项限制范围:
filter_var("200", FILTER_VALIDATE_INT, ["options" => ["min_range" => 0, "max_range" => 100]])→false
浮点数转整型要小心精度丢失和 .0 尾部
PHP 的 float 本质是 IEEE 754 双精度,很多小数无法精确表示。当你对 float 做 (int) 或 intval(),实际是先取整(向零截断),再丢掉小数部分——但前提是这个 float 本身没因精度问题“歪掉”。
典型坑:(int)(0.1 + 0.2) 不是 0,而是 0(因为 0.1+0.2 实际是 0.30000000000000004,截断后仍是 0),但 (int)(10000000000000000.1) 会变成 10000000000000000,因为 float 已经存不下小数位了。
- 不要依赖
(int)$float === $float来判断是否为整数——1e18这种大数即使显示为整数,内部也是 float,比较会失真 - 判断是否为“数学意义上整数”,该用
fmod($float, 1.0) === 0.0或is_int($var) || (is_float($var) && floor($var) === $var) - 涉及金额等敏感计算,从源头就该用字符串或整数分(如 100 表示 1 元),避免 float 中间态
(int) 看似省事,却常把格式错误悄悄吞掉。真正关键的地方,宁可多写一行 filter_var(),也别靠强制转换蒙混过关。











