PHP浮点数(float)运算过程中出现的错误问题解决方案


PHP开发中经常会涉及到金额的计算,而大多数线上项目的金额会精确到单位分,而在PHP运算中经常会遇到浮点数运算的偏差,导致金额统计错误和运算判断上的偏差。

案例1(浮点数相加):

$x = 0.1;
$y = 0.2;
$z = $x + $y;
var_dump($z); // 输出 float(0.3)

按照正常流程走下来,打印出0.3貌似根本没有问题,但是继续运算对比下:

var_dump($z==0.3); // 输出了 bool(false)

what?why?

案例1(浮点数相减):

$a = 70.6;
$b = 70.0;
$c = $a - $b;
var_dump($c); // 输出 float(0.59999999999999)

跟我们预想的输出结果貌似不一样?

这时候在某些商城商品购买或者退货退钱的情况下就会出现很多问题。

可能很多人会采用四舍五入或者保留N位小数来计算,某些情况下也是可以的,但其实PHP自身提供了一种高精度计算的函数。

BC Math高精确度的数学扩展,它可以为任意精度数学计算提供了二进制计算器(Binary Calculator),它支持任意大小和精度的数字,以字符串形式描述。在需要处理数字计算时,不要在简单地使用四则运算,而要用BC Math相关的函数来处理。

1、安装BC Math扩展

本类函数仅在 PHP 编译时配置了 –enable-bcmath 时可用。PHP 的 Windows 版本已内建对此扩展的支持,不需要载入额外的扩展来使用这些函数。如果需要编译安装,请参考PHP安装编译配置里的扩展 。

2、BC Math提供的函数

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个任意精度数字的减法

函数使用:

比较

/**
* 两个高精度数比较
* 
* @access global
* @param float $left
* @param float $right
* @param int $scale 精确到的小数点位数
* 
* @return int $left==$right 返回 0 | $left<$right 返回 -1 | $left>$right 返回 1
*/
 var_dump(bccomp($left=4.45, $right=5.54, 2));
 // -1

相加

/**
* 两个高精度数相加
* 
* @access global
* @param float $left
* @param float $right
* @param int $scale 精确到的小数点位数
*
* @return string
*/
var_dump(bcadd($left=1.0321456, $right=0.0243456, 2));
// 1.04

相减

/**
* 两个高精度数相减
* 
* @access global
* @param float $left
* @param float $right
* @param int $scale 精确到的小数点位数
* 
* @return string 
*/
 var_dump(bcsub($left=1.0321456, $right=3.0123456, 2));
 // -1.98

相除

/**
* 两个高精度数相除
* 
* @access global
* @param float $left
* @param float $right
* @param int $scale 精确到的小数点位数
* 
* @return string 
*/
 var_dump(bcdiv($left=6, $right=5, 2));
 // 1.20

相乘

/**
* 两个高精度数相乘
* 
* @access global
* @param float $left
* @param float $right
* @param int $scale 精确到的小数点位数
* 
* @return string 
*/
var_dump(bcmul($left=3.1415926, $right=2.4569874566, 2));
// 7.71

设置小数点

/**
* 设置bc函数的小数点位数
* 
* @access global
* @param int $scale 精确到的小数点位数
* 
* @return void 
*/
bcscale(3);
var_dump(bcdiv('105', '6.55957'));
// 16.007

声明:版权所有,违者必究|如未注明,均为原创|本网站采用 BY-NC-SA协议进行授权

转载:转载请注明原文链接 - PHP浮点数(float)运算过程中出现的错误问题解决方案

点击排行