如果用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