PHP浮点数运算不准确的问题

echo intval(0.58*100);//结果为57

echo intval((0.1 + 0.7) * 10);//结果为7

产生这样的原因是计算机内部对部分浮点数不能准确地用二进制表示,就像我们不能用十进制准确表示10/3一样.

浮点数在计算机内部的表示:IEEE 754.不懂的自己查找资料

也可以参考一下鸟哥这篇文章:http://www.laruence.com/2013/03/26/2884.html

<?php
echo intval(0.58*100);	//57,注意啦
echo '<br/>';
echo intval(bcmul(0.58, 100));	//58
echo '<br/>';
echo floor(strval(0.58*100));	//58
echo '<br/>';
echo (int)(0.58 * 1000/10);	//58
echo '<br/>';
echo intval((0.1 + 0.7) * 10);	//7,注意啦
echo '<br/>';
echo intval(bcadd("0.1", "0.7",1) * 10);  //8
?>

再看一个例子

<?php
if((0.1 + 0.7) == 0.8){
	echo '相等'; 
}else{
	echo '不相等'; //这里输出
}
echo '<br />';
$a = 0.1 + 0.7;
echo $a; //0.8
echo '<br/>';
if($a == 0.8){ 
	echo '一天一小步';
}else{
	echo '一年一大步'; //这里输出
}
echo '<br/>';
if(strval($a) == 0.8){
	echo '一天一小步'; //这里输出
}else{
	echo '一年一大步'; 
}
echo '<br/>';
if(bcadd(0.1, 0.7,1) == 0.8){
	echo '一天一小步'; //这里输出
}else{
	echo '一年一大步';
}
/*
结果:
不相等
0.8
一年一大步
一天一小步
一天一小步
*/
?>

$a = 0.00...00(n个0)1 + 0.7;用strval对$a进行取值的时候,会因为n个0的个数不同,导致$a和0.8是否相等的结果不同

而bcadd函数会对2个数相加,根据你选择保留几个小数,而舍弃后面的小数

所以对于浮点数来计算金额这些敏感的数据,推荐使用PHP的BC函数

BCMath 任意精度数学

  • bcadd — 2个任意精度数字的加法计算
  • bccomp — 比较两个任意精度的数字
  • bcdiv — 2个任意精度的数字除法计算
  • bcmod — 对一个任意精度数字取模
  • bcmul — 2个任意精度数字乘法计算
  • bcpow — 任意精度数字的乘方
  • bcpowmod — Raise an arbitrary precision number to another, reduced by a specified modulus
  • bcscale — 设置所有bc数学函数的默认小数点保留位数
  • bcsqrt — 任意精度数字的二次方根
  • bcsub — 2个任意精度数字的减法


猜你喜欢

转载自blog.csdn.net/weixin_37909391/article/details/80911087