
PHP浮点数转整型为什么经常丢精度
因为PHP底层用C的double类型存浮点数,二进制无法精确表示很多十进制小数(比如0.1),转整型前哪怕只是var_dump(0.1 + 0.2)都可能输出0.30000000000000004。直接(int)或intval()会直接截断小数部分,不四舍五入,也不处理隐式精度误差。
floor()、ceil()、round()、(int)到底怎么选
关键看你要的是“截断”、“向下取整”、“向上取整”,还是“四舍五入”。它们对负数行为完全不同,且round()默认使用“四舍六入五成双”规则(银行家舍入),不是小学数学里的四舍五入。
-
(int)和intval():纯截断,(int)3.9→3,(int)-3.9→-3(注意:不是-4) -
floor():向下取整,floor(3.9)→3,floor(-3.9)→-4 -
ceil():向上取整,ceil(3.1)→4,ceil(-3.1)→-3 -
round($f, 0, PHP_ROUND_HALF_UP):强制“传统四舍五入”,round(2.5, 0, PHP_ROUND_HALF_UP)→3;不传第三个参数时,round(2.5)可能得2(取决于PHP版本和平台)
用round()前必须先处理浮点误差
直接round(0.1 + 0.2)可能得0而不是1,因为0.1 + 0.2实际是0.30000000000000004,round()按默认精度(0位)会四舍五入到0——但你本意是算0.3再取整。
- 安全做法:先用
round($f, 14)把浮点误差抹掉(14位足够覆盖double精度极限),再二次取整 - 更稳妥:用
bcadd()做高精度加减,再转整,比如intval(bcadd('0.1', '0.2', 1)) - 避免场景:钱数计算绝不用
(int)或裸round(),必须用bcmul()/bcadd()配合固定小数位
PHP 8.1+ 的intdiv()和is_int()不能替代取整逻辑
intdiv()只用于两整数相除并向下取整,输入非整数会报TypeError;is_int()只判断类型,对3.0返回false——它根本不是float,但var_dump(3.0)显示float(3)。别指望靠类型判断绕过精度问题。
立即学习“PHP免费学习笔记(深入)”;
真正要命的坑是:你以为$x === 5.0就能当整数用,结果json_encode(['id' => $x])输出"id":5.0,下游JS解析成number但语义已变;这时候得用(int)$x或round($x)显式转,且确保$x本身没残留浮点误差。









