从事金融行业,资金运算频繁,这里说下我遇到的坑....稍不留神,用户资金可能损失几十万,甚至更可怕......直接上实例吧:

javascript

  1. 0.1 + 0.2 为啥不等于 0.3 ? (正确结果:0.30000000000000004
  2.  
  3. 0.8 * 7 为啥不等于 5.6 ? (正确结果:5.6000000000000005

PHP

  1. var_dump(intval(0.58 * 100));

正确结果是 57,而不是 58

浮点运算惹的祸

其实这些结果都并非语言的 bug,但和语言的实现原理有关, js 所有数字统一为 Number, 包括整形实际上全都是双精度(double)类型。

而PHP会区分 int 还是 float。不管什么语言,只要涉及浮点运算,都是存在类似的问题,使用时一定要注意。

  1. 说明:如果用php的+-*/计算浮点数的时候,可能会遇到一些计算结果错误的问题,比如上面 echo intval( 0.58*100 );会打印57,而不是58,这个其实是计算机底层二进制无法精确表示浮点数的一个bug,是跨语言的,我用python也遇到这个问题。所以基本上大部 分语言都提供了精准计算的类库或函数库,比如phpBC高精确度函数库,稍后我绍一下一些常用的BC高精确度函数使用。

还是回到上面的57,58问题。

为啥输出是57啊? PHP的bug么?

  要搞明白这个原因, 首先我们要知道浮点数的表示(IEEE 754):

  浮点数, 以64位的长度(双精度)为例, 会采用1位符号位(E), 11指数位(Q), 52位尾数(M)表示(一共64位).

  符号位:最高位表示数据的正负,0表示正数,1表示负数。

  指数位:表示数据以2为底的幂,指数采用偏移码表示

  尾数:表示数据小数点后的有效数字.

  这里的关键点就在于, 小数在二进制的表示, 关于小数如何用二进制表示, 大家可以百度一下, 我这里就不再赘述, 我们关键的要了解, 0.58 对于二进制表示来说, 是无限长的值(下面的数字省掉了隐含的1)..

  0.58的二进制表示基本上(52位)是: 00101000111101011100001010001111010111000010100011110.57的二进制表示基本上(52位)是: 001000111101011100001010001111010111000010100011110而两者的二进制, 如果只是通过这52位计算的话,分别是:www.111cn.net

  0.58 -> 0.579999999999999960.57 -> 0.5699999999999999至于0.58 * 100的具体浮点数乘法, 我们不考虑那么细, 有兴趣的可以看(Floating point), 我们就模糊的以心算来看… 0.58 * 100 = 57.999999999

  那你intval一下, 自然就是57了….

  可见, 这个问题的关键点就是: “你看似有穷的小数, 在计算机的二进制表示里却是无穷的”

  因此, 不要再以为这是PHP的bug了, 这就是这样的…..

  PHP浮点型在进行+-*%/存在不准确的问题

继续看一段代码:

  1. $a = 0.1;
  2. $b = 0.7;
  3. var_dump(($a + $b) == 0.8); // false

打印出来的值为 boolean false

  这是为啥?PHP手册对于浮点数有以下警告信息:

  Warning

  浮点数精度

  显然简单的十进制分数如同 0.1 或 0.7 不能在不丢失一点点精度的情况下转换为内部二进制的格式。这就会造成混乱的结果:例如,floor((0.1+0.7)*10) 通常会返回 7 而不是预期中的 8,因为该结果内部的表示其实是类似 7.9999999999…。

  这和一个事实有关,那就是不可能精确的用有限位数表达某些十进制分数。例如,十进制的 1/3 变成了 0.3333333. . .。

  所以永远不要相信浮点数结果精确到了最后一位,也永远不要比较两个浮点数是否相等。如果确实需要更高的精度,应该使用任意精度数学函数或者 gmp 函数

那么上面的算式我们应该改写为

  1. $a = 0.1;
  2. $b = 0.7;
  3. var_dump(bcadd($a,$b,2) == 0.8); // true

常用的高精度函数如下:

  1.   bcadd 将两个高精度数字相加
  2.  
  3.   bccomp 比较两个高精度数字,返回-1, 0, 1
  4.  
  5.   bcdiv 将两个高精度数字相除
  6.  
  7.   bcmod 求高精度数字余数
  8.  
  9.   bcmul 将两个高精度数字相乘
  10.  
  11.   bcpow 求高精度数字乘方
  12.  
  13.   bcpowmod 求高精度数字乘方求模,数论里非常常用
  14.  
  15.   bcscale 配置默认小数点位数,相当于就是Linux bc中的”scale=”
  16.  
  17.   bcsqrt 求高精度数字平方根
  18.  
  19.   bcsub 将两个高精度数字相减
  1. BC高精确度函数库包含了:相加,比较,相除,相减,求余,相乘,n次方,配置默认小数点数目,求平方。这些函数在涉及到有关金钱计算时比较有用,比如电商的价格计算。
  1. /**
  2. * 两个高精度数比较
  3. *
  4. * @access global
  5. * @param float $left
  6. * @param float $right
  7. * @param int $scale 精确到的小数点位数
  8. *
  9. * @return int $left==$right 返回 0 | $left<$right 返回 -1 | $left>$right 返回 1
  10. */
  11. var_dump(bccomp($left=4.45, $right=5.54, 2));
  12. // -1
  13.  
  14. /**
  15. * 两个高精度数相加
  16. *
  17. * @access global
  18. * @param float $left
  19. * @param float $right
  20. * @param int $scale 精确到的小数点位数
  21. *
  22. * @return string
  23. */
  24. var_dump(bcadd($left=1.0321456, $right=0.0243456, 2));
  25. //1.05
  26.  
  27. /**
  28. * 两个高精度数相减
  29. *
  30. * @access global
  31. * @param float $left
  32. * @param float $right
  33. * @param int $scale 精确到的小数点位数
  34. *
  35. * @return string
  36. */
  37. var_dump(bcsub($left=1.0321456, $right=3.0123456, 2));
  38. //-1.98
  39.  
  40. /**
  41. * 两个高精度数相除
  42. *
  43. * @access global
  44. * @param float $left
  45. * @param float $right
  46. * @param int $scale 精确到的小数点位数
  47. *
  48. * @return string
  49. */
  50. var_dump(bcdiv($left=6, $right=5, 2));
  51. //1.20
  52.  
  53. /**
  54. * 两个高精度数相乘
  55. *
  56. * @access global
  57. * @param float $left
  58. * @param float $right
  59. * @param int $scale 精确到的小数点位数
  60. *
  61. * @return string
  62. */
  63. var_dump(bcmul($left=3.1415926, $right=2.4569874566, 2));
  64. //7.71
  65.  
  66. /**
  67. * 设置bc函数的小数点位数
  68. *
  69. * @access global
  70. * @param int $scale 精确到的小数点位数
  71. *
  72. * @return void
  73. */
  74. bcscale(3);
  75. var_dump(bcdiv('105', '6.55957'));
  76. //php7.1 16

封装方法:

  1. /**
  2.  
  3. * 精确加法
  4.  
  5. * @param [type] $a [description]
  6.  
  7. * @param [type] $b [description]
  8.  
  9. */
  10.  
  11. function math_add($a,$b,$scale = '2') {
  12.  
  13. return bcadd($a,$b,$scale);
  14.  
  15. }
  16.  
  17. /**
  18.  
  19. * 精确减法
  20.  
  21. * @param [type] $a [description]
  22.  
  23. * @param [type] $b [description]
  24.  
  25. */
  26.  
  27. function math_sub($a,$b,$scale = '2') {
  28.  
  29. return bcsub($a,$b,$scale);
  30.  
  31. }
  32.  
  33. /**
  34.  
  35. * 精确乘法
  36.  
  37. * @param [type] $a [description]
  38.  
  39. * @param [type] $b [description]
  40.  
  41. */
  42.  
  43. function math_mul($a,$b,$scale = '2') {
  44.  
  45. return bcmul($a,$b,$scale);
  46.  
  47. }
  48.  
  49. /**
  50.  
  51. * 精确除法
  52.  
  53. * @param [type] $a [description]
  54.  
  55. * @param [type] $b [description]
  56.  
  57. */
  58.  
  59. function math_p($a,$b,$scale = '2') {
  60.  
  61. return bcp($a,$b,$scale);
  62.  
  63. }
  64.  
  65. /**
  66.  
  67. * 精确求余/取模
  68.  
  69. * @param [type] $a [description]
  70.  
  71. * @param [type] $b [description]
  72.  
  73. */
  74.  
  75. function math_mod($a,$b) {
  76.  
  77. return bcmod($a,$b);
  78.  
  79. }
  80.  
  81. /**
  82.  
  83. * 比较大小
  84.  
  85. * @param [type] $a [description]
  86.  
  87. * @param [type] $b [description]
  88.  
  89. * 大于 返回 1 等于返回 0 小于返回 -1
  90.  
  91. */
  92.  
  93. function math_comp($a,$b,$scale = '5') {
  94.  
  95. return bccomp($a,$b,$scale); // 比较到小数点位数
  96.  
  97. }

.

php高精度计算问题的更多相关文章

  1. php-浮点数计算,double类型数加减乘除必须用PHP提供的高精度计算函数

    一.前方有坑 php在使用加减乘除等运算符计算浮点数的时候,经常会出现意想不到的结果,特别是关于财务数据方面的计算,给不少工程师惹了很多的麻烦.比如今天工作终于到的一个案例: $a = 2586; $ ...

  2. 使用高精度计算斐波那契数列 c++

    使用高精度计算斐波那契数列 非高精度 Code(Non-high accuracy) 这是不用高精度的代码 #include<bits/stdc++.h> using namespace ...

  3. 1169A+B问题终结版(高精度计算)

    描述 给定两个整数A和B,输出A+B的值.A和B的值可能会很大很大,甚至达到100位.现在请你解决这一问题. 输入 两行,分别是两个整数A,B,换行隔开.A和B会很大很大. 输出 一个整数,即A+B的 ...

  4. C++高精度计算代码运行时间(转载)

    转载:http://blog.csdn.net/rrrfff/article/details/6583410 //在定时前应该先调用QueryPerformanceFrequency()函数获得机器内 ...

  5. 高精度计算的类(BigInteger和BigDecimal)

    这两个类 在Java中没有对应的基本类型.不过,这两个类包含的方法,提供的操作与对基本类型所能执行的操作差不多. 也就是说,能对基本类型 int float 等的操作,也同样能作用于这两个类,只不过必 ...

  6. HDU 1042 N!(高精度计算阶乘)

    N! Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submiss ...

  7. uva424 高精度计算

    题意是计算各长整数的和,最多输入100个数,每个数都是正整数且位数不超过100. 很明显100位的数用数组存会方便许多,然后设置两个整型数组,一个存进位,一个存结果.为了对齐进行运算,我将所有的数先逆 ...

  8. BigDecimal 高精度计算 熟悉扩展,java除法保留小数问题

    java保留两位小数问题: 方式一: 四舍五入  double   f   =   111231.5585;  BigDecimal   b   =   new   BigDecimal(f);  d ...

  9. 信息学奥赛一本通算法(C++版)基础算法:高精度计算

    高精度加法(大位相加) #include <bits/stdc++.h> using namespace std; int main() { ],b1[]; ],b[],c[];//a,b ...

  10. Java 使用BigDecimal类处理高精度计算

    Java在java.math包中提供的API类BigDecimal,用来对超过16位有效位的数进行精确的运算.双精度浮点型变量double可以处理16位有效数,但在实际应用中,可能需要对更大或者更小的 ...

随机推荐

  1. kafka创建topics 错误: 找不到或无法加载主类 Files\Java\jdk1.7.0_80\lib;C:\Program

    解决方案如下: 在kafka安装目录中找到bin\windows目录中的kafka-run-class.bat找到%CLASSPATH%为其加上双引号

  2. 电子产品使用感受之——为什么我那么喜欢2015年的11.6寸MacBook Air?

    2018年Mac笔记本产品线得到了一次更新,Mac book Pro, MacBook Air更新后的Mac 产品线变得更加让人摸不着头脑,价格昂贵不说,产品分类细化到如此程度,让一个选择困难症的消费 ...

  3. mongodb 参数优化

    1.大部分IO操作为随机IO,建议采用SSD或PCIE,普通硬盘RAID10 2.IO调度算法.普通磁盘:deadline避免IO请求出现“饥饿”现象,SSD/PCIE:noop简单的先进先出处理请求 ...

  4. subline text3 安装 rem装换工具

    CSSREM 一个CSS的px值转rem值的Sublime Text 3自动完成插件. 插件效果如下: 安装 下载本项目,比如:git clone https://github.com/flashli ...

  5. sourceTree如何不用注册就使用

    下载好之后会有这么一个界面要求你注册或登录.(不管它)将下面的一串串放进我的电脑的地址栏,打开sourcetree的文件夹 %LocalAppData%\Atlassian\SourceTree\ 注 ...

  6. Xposed免重启调试工具类

    直接放代码 package com.xirtam.hello; import android.app.Application; import android.content.Context; impo ...

  7. vue框架与koa2服务器实现跨域通信

    首先我们在vue中引入axios, npm install axios --save 在需要用到的页面引入axios import axios from "axios"; 用axi ...

  8. Mockito单元测试实战

    最近使用Mockito完成了几个简单的测试,写个博客mark一下: 第一种模拟web请求 @SpringBootTest @RunWith(SpringRunner.class) @WebAppCon ...

  9. Vue学习记录第二天

    又来做笔记啦,今天又自暴自弃了,还好及时清醒过来了,什么时候努力都不晚,主要是要一直坚持下去,只要坚持就一定会有收获,所有成功得人背后都是付出了巨大得努力的,没有人平白无故的成功.看似光鲜亮丽的背后, ...

  10. MySQL 误删数据、误更新数据(update,delete忘加where条件)

    MySQL 误操作后数据恢复(update,delete忘加where条件) 关键词:mysql误删数据,mysql误更新数据 转自:https://www.cnblogs.com/gomysql/p ...