標籤:alt oid ret 列印 man 浮點 管理系統 函數 highlight
本文執行個體講述了PHP中兩個float(浮點數)比較方法。分享給大家供大家參考。具體如下:
最近在開發一個合約管理系統的時候,涉及到兩個浮點數比較,算是把我鬱悶慘了。
在N久以前,就不曉得從哪裡聽來的一個“不要用等號去比較浮點數”的“真理”,自己平時也在用,好像沒有出現啥問題,可這次問題總算是來了。
?
| 123456789 |
<?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,小數位)來比較。
?
| 123456789 |
<?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 返回 110 */11 var_dump(bccomp($left=4.45, $right=5.54, 2));12 // -113 14 /**15 * 兩個高精度數相加16 * 17 * @access global18 * @param float $left19 * @param float $right20 * @param int $scale 精確到的小數點位元21 * 22 * @return string 23 */24 var_dump(bcadd($left=1.0321456, $right=0.0243456, 2));25 //1.0426 27 /**28 * 兩個高精度數相減29 * 30 * @access global31 * @param float $left32 * @param float $right33 * @param int $scale 精確到的小數點位元34 * 35 * @return string 36 */37 var_dump(bcsub($left=1.0321456, $right=3.0123456, 2));38 //-1.9839 40 /**41 * 兩個高精度數相除42 * 43 * @access global44 * @param float $left45 * @param float $right46 * @param int $scale 精確到的小數點位元47 * 48 * @return string 49 */50 var_dump(bcdiv($left=6, $right=5, 2));51 //1.2052 53 /**54 * 兩個高精度數相乘55 * 56 * @access global57 * @param float $left58 * @param float $right59 * @param int $scale 精確到的小數點位元60 * 61 * @return string 62 */63 var_dump(bcmul($left=3.1415926, $right=2.4569874566, 2));64 //7.7165 66 /**67 * 設定bc函數的小數點位元68 * 69 * @access global70 * @param int $scale 精確到的小數點位元71 * 72 * @return void 73 */ 74 bcscale(3);75 var_dump(bcdiv(‘105‘, ‘6.55957‘)); 76 // 16.007
php浮點數比較