php浮点数比较

Posted 防空洞123

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了php浮点数比较相关的知识,希望对你有一定的参考价值。

本文实例讲述了php中两个float(浮点数)比较方法。分享给大家供大家参考。具体如下:

最近在开发一个合同管理系统的时候,涉及到两个浮点数比较,算是把我郁闷惨了。
在N久以前,就不晓得从哪里听来的一个“不要用等号去比较浮点数”的“真理”,自己平时也在用,好像没有出现啥问题,可这次问题总算是来了。

1
2
3
4
5
6
7
8
9
<?php
$sum = "12300.00";
$a  = "10000.30";
$b  = "2000.30";
$c  "299.40";
$sum = (float) $sum;
$s = (float) ($a+$b+$c);
var_dump($sum, $s);
var_dump($sum==$s);

结果是:

float(12300)
float(12300)
bool(false)

后来才知道在PHP中,要比较两个浮点数的大小,可以用bccomp(参数1,参数2,小数位)来比较。

1
2
3
4
5
6
7
8
9
<?php
$sum = "12300.00";
$a  = "10000.30";
$b  = "2000.30";
$c  "299.40";
$sum = (float) $sum;
$s = (float) ($a+$b+$c);
var_dump($sum, $s);
var_dump(bccomp($sum,$s,2));

结果:

float(12300)
float(12300)
int(0) // 0表示两个浮点数值相等

<?php

echo bccomp(‘1‘, ‘2‘) . "\n";   // -1 小于
echo bccomp(‘1.00001‘, ‘1‘, 3); // 0 等于
echo bccomp(‘1.00001‘, ‘1‘, 5); // 1 大于

?>

 

如果用php的+-*/计算浮点数的时候,可能会遇到一些计算结果错误的问题,比如echo intval( 0.58*100 );会打印57,而不是58

这个其实是计算机底层二进制无法精确表示浮点数的一个bug,是跨语言的

可以用精度函数库解决问题

  

  bcadd — 将两个高精度数字相加

  bccomp — 比较两个高精度数字,返回-1, 0, 1

  bcdiv — 将两个高精度数字相除

  bcmod — 求高精度数字余数

  bcmul — 将两个高精度数字相乘

  bcpow — 求高精度数字乘方

  bcpowmod — 求高精度数字乘方求模,数论里非常常用

  bcscale — 配置默认小数点位数,相当于就是Linux bc中的”scale=”

  bcsqrt — 求高精度数字平方根

  bcsub — 将两个高精度数字相减

 

技术分享
 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.04
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 // 16.007

以上是关于php浮点数比较的主要内容,如果未能解决你的问题,请参考以下文章

php随机浮点数都有哪些?比如从0.1到3.0中随机一个浮点数出来?

php浮点数比较不相等的问题

OpenGL:为啥我不能将单个浮点数从顶点着色器传递到片段着色器?

php浮点数加减乘除bug

php浮点数

原生JavaScript判断是否为邮箱危险字符验证长度验证网址验证小数整数浮点数等常用的验证