float php 运算_系统的讲解 - PHP 浮点数高精度运算

概述

记录下,工作中遇到的坑 ...

关于 PHP 浮点数运算,特别是金融行业、电子商务订单管理、数据报表等相关业务,利用浮点数进行加减乘除时,稍不留神运算结果就会出现偏差,轻则损失几十万,重则会有信誉损失,甚至吃上官司,我们一定要引起高度重视!

浮点数运算的“锅”

//加

$a = 0.1;

$b = 0.7;

$c = intval(($a + $b) * 10);

echo $c."
";

//输出:7

//减

$a = 100;

$b = 99.98;

$c = $a - $b;

echo $c."
";

//输出:0.019999999999996

//乘

$a = 0.58;

$b = 100;

$c = intval($a * $b);

echo $c."
";

//输出:57

//除

$a = 0.7;

$b = 0.1;

$c = intval($a / $b);

echo $c."
";

//输出:6

上面的结果,显然不是我们想要的!

PHP 官方手册解释如下:

浮点数的精度有限。尽管取决于系统,PHP 通常使用 IEEE 754 双精度格式,则由于取整而导致的最大相对误差为 1.11e-16。非基本数学运算可能会给出更大误差,并且要考虑到进行复合运算时的误差传递。永远不要相信浮点数结果精确到了最后一位,也永远不要比较两个浮点数是否相等。如果确实需要更高的精度,应该使用任意精度数学函数 或者 gmp 函数。

这里的关键在于,浮点数的小数用二进制的表示,过程如下:

将小数乘以2,取整数部分表示第一位;

将小数部分乘以2,取整数部分表示第二位;

再将小数部分乘以2,取整数部分表示第三位;

... 依次类推,直到小数部分为0;

例:0.58

0.58 * 2 = 1.16 ---> 1

0.16 * 2 = 0.32 ---> 0

0.32 * 2 = 0.64 ---> 0

0.64 * 2 = 1.28 ---> 1

0.28 * 2 = 0.56 ---> 0

0.56 * 2 = 1.12 ---> 1

0.12 * 2 = 0.24 ---> 0

0.24 * 2 = 0.48 ---> 0

0.48 * 2 = 0.96 ---> 0

0.96 * 2 = 1.92 ---> 1

...

我们会得到一个无限循环的二进制小数:

0.1001010001...

小数部分出现循环,有限的二进制位无法准确的表示一个小数,这也就是小数运算出现误差的原因。

接下来给大家介绍 任意精度数学函数。

任意精度数学函数

对于任意精度的数学,PHP 提供了支持用字符串表示的任意大小和精度的数字的二进制计算。

BCMath:BC 是 Binary Calculator 的缩写。

大家在使用前,请先确认是否已安装 bcmath。

//加

$a = 0.1;

$b = 0.7;

$c = intval(bcadd($a, $b, 1) * 10);

echo $c."
";

//输出:8

//减

$a = 100;

$b = 99.98;

$c = bcsub($a, $b, 2);

echo $c."
";

//输出:0.02

//乘

$a = 0.58;

$b = 100;

$c = intval(bcmul($a, $b));

echo $c."
";

//输出:58

//除

$a = 0.7;

$b = 0.1;

$c = intval(bcdiv($a, $b));

echo $c."
";

//输出:7

除了加减乘除,bcmath 还提供了以下方法:

bccomp 比较两个任意精度的数字

bcmod 对一个任意精度数字取模

bcpow 任意精度数字的乘方

bcpowmod 高精度数字乘方求模

bcscale 设置所有bc数学函数的默认小数点保留位数

bcsqrt 任意精度数字的二次方根

常用数值处理方案

舍去法取整(向下取整)

echo floor(5.1);

//输出:5

echo floor(8.8);

//输出:8

进一法取整(向上取整)

echo ceil(5.1);

//输出:6

echo ceil(8.8);

//输出:9

普通四舍五入法

echo round(5.1);

//输出:5

echo round(8.8);

//输出:9

//保留两位小数并且进行四舍五入

echo round(5.123, 2);

//输出:5.12

echo round(8.888, 2);

//输出:8.89

//保留两位小数并且不进行四舍五入

echo substr(round(5.12345, 3), 0, -1);

//输出:5.12

echo substr(round(8.88888, 3), 0, -1);

//输出:8.88

银行家舍入法

四舍六入五考虑,五后非空就进一,五后为空看奇偶,五前为偶应舍去,五前为奇要进一。

保留两位小数,例:

1.2849 = 1.28 -> 四舍

1.2866 = 1.29 -> 六入

1.2851 = 1.29 -> 五后非空就进一

1.2850 = 1.28 -> 五后为空看奇偶,五前为偶应舍去

1.2750 = 1.28 -> 五后为空看奇偶,五前为奇要进一

实现代码如下:

echo round(1.2849, 2, PHP_ROUND_HALF_EVEN);

//输出:1.28

echo round(1.2866, 2, PHP_ROUND_HALF_EVEN);

//输出:1.29

echo round(1.2851, 2, PHP_ROUND_HALF_EVEN);

//输出:1.29

echo round(1.2850, 2, PHP_ROUND_HALF_EVEN);

//输出:1.28

echo round(1.2750, 2, PHP_ROUND_HALF_EVEN);

//输出:1.28

更多 round 使用说明,请查阅官方手册:

数值格式化(千位分组)

应用于金额的展示,比如我们经常会看的银行卡余额。

echo number_format('10000.98', 2, '.', ',');

//输出:10,000.98

echo number_format('340888999', 2, '.', ',');

//输出:340,888,999.00

扩展

MySQL 浮点型字段

在 MySQL 中,创建表字段时也有浮点数类型。

浮点数类型包括单精度浮点数(float)和双精度浮点数(double)。

同理,不建议使用浮点数类型!!!

浮点数存在误差,当我们使用精度敏感的数据时,应该使用定点数(decimal)进行存储。

小结

通过浮点数精度的问题,了解到浮点数的小数用二进制的表示。

分享了用 PHP 任意精度数学函数,来进行高精度运算。

同时分享了常用数值处理方案,比如舍去法、进一法、四舍五入法、银行家舍入法、数值格式化 等。

最后,通过 PHP 的 float 联想到 MySQL 的 float。

以后,在使用浮点数运算的时候,一定要慎之又慎,细节决定成败。

本文欢迎转发,转发请注明作者和出处,谢谢!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/488798.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

20155234 2016-2017-2 《Java程序设计》第5周学习总结

20155234 2016-2017-2 《Java程序设计》第5周学习总结 教材学习内容总结 Java中所有错误都会被打包为对象,运用try、catch,可以在错误发生时显示友好的错误信息。运用try、catch,还可以在捕捉处理错误之后,尝试恢复程序正常执行流…

harmonyos 编译打包生成app,鸿蒙(HarmonyOS)App开发入门(2)—运行HelloWorld

前言为确保HarmonyOS应用的完整性,HarmonyOS通过数字证书和授权文件来对应用进行管控,只有签名过的HAP(编译产物,后面章节详细讲述,本章主要讲证书申请到代码运行到真机)才允许安装到设备上进行运行(如果不带签名信息,…

【周末阅读】你不得不知道的10大AI赋能行业,你身边都能找到!

来源:青岛智能产业技术研究院智能产业 前沿高地【导读】AI正突破诸多技术瓶颈,在市政、交通、医疗、等行业横向渗透,逐步走入大众日常生活中。国务院印发的《新一代人工智能发展规划》中指出,2020年人工智能产业规模将超过1500亿元…

bindresult必须在哪个位置_管道疏通剂哪个牌子好 管道疏通机使用方法有哪些

平时大家不用的水或者一些物品,在处理的时候应该都会倒到下水道之中,而下水道确实具备着这一种效果,但很多时候,下水道往往会因为口比较小,而被一些物品所堵塞,这样一来,影响上其实会非常大&…

js 递归查询所有的叶子结点_json树递归js查询json父子节点

上代码,直接另存为html打开,看console控制台就可以看到效果了json-queryvar json [{"MenuCode": "S0-3100-00","MenuAction": "TelPay","MenuName": "通讯费","subMenu": [{&qu…

html有序列表和无序列表互相嵌套,关于列表的嵌套,下列说法正确的是( )。 答案:无序列表和有序列表可以相互嵌套与 之间相当于一个容器,可以嵌套无序列表、有序列表等网页元素...

相关问题危险化学品不易存放高处,避免取用时容器坠落发生意外: 化学品 高处 时 容器 意外密封放射源是指:: 密封放射源 研究堆 动力 核燃料 循环 范畴 材料 以外 容器 中 层 固态 放射性 材料以下控件中不属于容器的是( )。: 以下 控件 中 容器 Image容器…

华为深度学习新模型DeepShift:移位和求反代替乘法,神经网络成本大降

来源:机器学习研究会订阅号【导读】深度学习模型,尤其是卷积神经网络的计算成本问题主要是由于卷积层和全连接层中大量进行乘法运算造成的。华为异构实验室的研究人员提出,用移位和求反运算代替乘法,可有效缓解计算成本过高的问题…

如何把竖排的数据变为横排_如何有效的自学日语自学的经验都在这里了

一.基础知识引入自学一门语言之前,先要了解这门语言的基本信息。从狭义上来说,语言是能够传达意义的声音,每个语种都由一定数量的声音构成。比如日语,日语普通话中的声音一共有104个。当这些声音对应成符号书写下来时&…

dir结构体 linux_Linux下DIR,dirent,stat等结构体详解

最近在看Linux下文件操作相关章节,遇到了这么几个结构体,被搞的晕乎乎的,今日有空,仔细研究了一下,受益匪浅。首先说说DIR这一结构体,以下为DIR结构体的定义:struct__dirstream{void*__fd;char*…

447. Number of Boomerangs

题目: Given n points in the plane that are all pairwise distinct, a "boomerang" is a tuple of points (i, j, k) such that the distance between iand j equals the distance between i and k (the order of the tuple matters). Find the number …

jquery获取html页面参数乱码,JS或Jquery获取浏览器URL的参数值 汉字值乱码 并转码...

1、获取url很简单,代码如下:window.location.href;2、jquery获取url参数比较复杂,要用到正则表达式,所以学好javascript正则式多么重要的事情首先看看单纯的通过javascript是如何来获取url中的某个参数://获取url中的参…

信阳农林技术学院经纬度_信阳无人机创业团队:让梦想在田野起飞

青春由磨砺而出彩,人生因奋斗而升华。在河南信阳,有这样一个年轻的创业团队,他们立志扎根农村做新型农民。他们在大学期间就钻研如何把无人机技术和现代农业生产相结合,并且运用到田间地头,在田野之上放飞青春梦想。一…

edge怎么开启沉浸式阅读_建安中艺光影秀|景区乐园开启沉浸式光影秀互动之旅...

随着时代的发展,文旅夜游的主旋律逐渐向声、光、电的科技和水、陆、空的视听体验靠近,越来越多的园林景点打造沉浸式光影互动投影秀,将历史人文与自然界的静谧纯粹相结合,利用沉浸式光影秀将园林景区中假山、悬崖、溶洞、峭壁、飞…

李德毅院士:自动驾驶与智能网联

来源:58车在电动化、智能化、网联化革命带动下,传统的汽车产业供应链正在重塑,中国电动汽车发展进入结构调整期。以“把握形势 聚焦转型 引领创新”为主题的2020中国电动汽车百人会论坛于1月10-12日在钓鱼台国宾馆隆重召开。秉持“传递权威信…

有用的Python模块 - pprint

当想在终端打印一个很大的字典或者一个很长的列表时,总是被print打印出来的效果气懵在电脑前,现在有pprint就不用担心啦。 最直接的使用方式就是 import pprint pprint.pprint(object) ex: import pprintd {user: icebug,gender: male }pprint.pprint(d…

android录音播放列表,android数据库里的视频,图片,音频表

数据库模式分析图片数据库图片数据库由两个表组成,分别是 images 和 thumbnails,物理数据模型如下所示(Power Designer 逆向工程生成)Note: 如何数据库物理模型图: 表示此为主键。其余的表名、字段名、数据类型应该都能看明白。Note: SQLite …

科技领域看不见的手如何影响企业未来命运

作者:刘锋“看不见的手”像幽灵一样盘踞在人类社会的发展过程中,时隐时现,在亚当斯密的《国富论》中,看不见的手推动了经济的发展;在达尔文的进化论中,看不见的手推动了生物的自然选择;同样在21…

电脑动态屏保_Mac待机也能看时间,你需要一款时钟屏保!

Mac待机也能看时间啦,时钟屏保软件就可以做到哦,今天就为大家分享几款,来看看有没有喜欢的!Mac待机时也能看时间(原文)​www.macz.com第一款:Word Clock 一起进入Word时钟的词云。 Word Clock M…

怎样用python搭建简单的系统_如何用Python搭建一个简单的推荐系统?

本文使用的数据集是MovieLens数据集,该数据集由明尼苏达大学的Grouplens研究小组整理。它包含1,10和2亿个评级。 Movielens还有一个网站,我们可以注册,撰写评论并获得电影推荐。接下来我们就开始实战演练。在这篇文章中,我们会使用…

垂直居中——登录界面

弹性盒子垂直居中&#xff1a;body直接display&#xff1a;flex不行中间在来一层。 1 <!DOCTYPE html>2 <html>3 <head>4 <meta charset"utf-8">5 <link rel"stylesheet" href"https://cdn.static.runoob.com/libs/bootst…