当前位置:首页 > php > PHP浮点数(float)运算过程中出现的错误问题解决方案

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

原创 2019-10-17 php 1201
分享给朋友:

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 协议进行授权

转载:转载请注明原文链接,违者必究 - :https://www.wolfcode.net/info/162/

分享给朋友:

相关文章

webman自定义进程

1、新建文件 process/Rpc.php 编写rpc进程&lt;?php namespace&nbsp;process; use&nbsp;Workerman\Connection\TcpConnection; class&nbsp;Rpc { &nbsp;&nbsp;&nbsp;&n

在PHP7+下监控Memcached服务、性能、扇区等信息

PHPMemcachedAdmin:一款网页可视化的Memcached工具

Swoole的PHP协程开发框架 imi 集成ThinkPHP模板引擎 think-template

在 imi 框架根目录中执行composer&nbsp;require&nbsp;topthink/think-template安装好模板引擎后,找到路径\vendor\topthink\think-template\src\Template.php找到 fetch() 渲染模板文件的方法,将最后一

PHP7下MongoDB自增或自减一个字段的值

findAndModify属于原子操作模型数据,所谓原子操作就是要么这个文档保存到Mongodb,要么没有保存到Mongodb,不会出现查询到的文档没有保存完整的情况。

PHP的面向对象解析

早期编程由于受电脑硬件限制,程序都是追求效率,而忽略可理解性,扩充性,随着硬件技术的发展,编程越来越重视多人开发,程序员越来越重视程序的可靠性,可扩展性,可维护性,所以刺激了程序语言的发展

ThinkPHP6.0使用EasyTask常驻内存多进程任务管理

PHP常驻内存的多进程任务管理器Composer包。以进程管理为出发点,同时也支持为每个进程设置定时执行功能,您可以用它来完成需要重复运行的任务(如订单超时自动取消,短信邮件异步推送,队列/消费者/频道订阅者等等),甚至处理计划任务。

PHP8新特性盘点

PHP 8.0.0 已经正式发布了,这个对于PHPer无疑是一个令人振奋的消息。它包含了很多新功能与优化项, 包括命名参数、联合类型、注解、构造器属性提升、match表达式、nullsafe运算符、JIT,并改进了类型系统、错误处理、语法一致性。

PHP8新特性系列:构造器属性提升使用及注意事项

本篇主要说下PHP8构造器属性提升的用法,这个特性对于一些需要在构造器中设置或初始化一些类属性的时候非常有用(包括public、protected和private),比如在PHP7中你可以这样定义一个类的属性,然后在构造方法中传值。class&nbsp;Point&nbsp;{ &nbsp;&nb

ThinkPHP6.0在PHP8下报错解决方法

PHP8下全新安装ThinkPHP6.0.X出现报错,如下Deprecated:&nbsp;Method&nbsp;ReflectionParameter::getClass()&nbsp;is&nbsp;deprecated&nbsp;in&nbsp;xxxx\vendor\topthink\fr

MongoDB驱动聚合查询aggregate在分组分页排序后出现的数据混乱问题

最近在获取MongoDB数据时需要把重复的数据分组来排序,语言版本:PHP7