本文實例講述了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

