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