在日常开发中,经常会涉及到浮点数比较,会出现看上去应该返回true,结果返回false的现象。比如:
var_dump(0.4-0.3==0.1);
运行 返回值
然后我们执行:
echo 0.4-0.3;
输出结果 0.1 。看上去完全没有问题。然而结果不是我们想要的,让人很是头疼。
其实,这就是二进制存储<这里不做详细讲解>的问题了。
下面介绍一个函数,借助函数 我们可以看到以上减法输出结果的‘真身’。
serialize <其实就是我们常见的序列化函数>
serialize
(PHP 4, PHP 5, PHP 7)
serialize — 产生一个可存储的值的表示
描述
string serialize ( mixed $value )
执行代码:
echo serialize(0.4-0.3);
输出 d:0.10000000000000003;
由此我们就可以非常清晰的看出 上述返回false的原因了。
那我们浮点数该如何比较呢?
1、转化成字符串进行比较,借助函数 : strval
var_dump(strval(0.4-0.3)==0.1);
2、借助浮点数比较大小函数
精度函数库函数
bcadd — 将两个高精度数字相加
bccomp — 比较两个高精度数字,返回-1, 0, 1
bcdiv — 将两个高精度数字相除
bcmod — 求高精度数字余数
bcmul — 将两个高精度数字相乘
bcpow — 求高精度数字乘方
bcpowmod — 求高精度数字乘方求模,数论里非常常用
bcscale — 配置默认小数点位数,相当于就是Linux bc中的”scale=”
bcsqrt — 求高精度数字平方根
bcsub — 将两个高精度数字相减
以 bccomp 为例
bccomp
(PHP 4, PHP 5, PHP 7)
bccomp — 比较两个任意精度的数字
说明
int bccomp ( string $left_operand , string $right_operand [, int $scale = int ] )
把right_operand和left_operand作比较, 并且返回一个整数的结果.
参数
left_operand
左边的运算数, 是一个字符串.
right_operand
右边的运算数, 是一个字符串.
scale
可选的scale参数被用作设置指示数字, 在使用来作比较的小数点部分.
返回值
如果两个数相等返回0, 左边的数left_operand比较右边的数right_operand大返回1, 否则返回-1.
示例:
$a = 0.4-0.3;
var_dump(bccomp($a, 0.1));
3、四舍五入小数点后指定位数比较<少用> round
示例:
var_dump(round($a, 2) == 0.1);