浮點數的精度有限。儘管取決於系統,PHP 一般使用 IEEE 754 雙精度格式,則因爲取整而致使的最大相對偏差爲 1.11e-16。非基本數學運算可能會給出更大偏差,而且要考慮到進行復合運算時的偏差傳遞。
此外,以十進制可以精確表示的有理數如 0.1 或 0.7,不管有多少尾數都不能被內部所使用的二進制精確表示,所以不能在不丟失一點點精度的狀況下轉換爲二進制的格式。這就會形成混亂的結果:例如,floor((0.1+0.7)*10) 一般會返回 7 而不是預期中的 8,由於該結果內部的表示實際上是相似 7.9999999999999991118…。
因此永遠不要相信浮點數結果精確到了最後一位,也永遠不要比較兩個浮點數是否相等。若是確實須要更高的精度,應該使用任意精度數學函數或者 gmp 函數。
php
$param['orderFee'] = ncPriceYuan2fen($order_pay_info['api_pay_amount']);
ncPriceCalculate 採用三個參數的方式,直接傳入+-符號來執行不一樣的方法,和策略模式有一點點像,我的認爲比起定義多個方法去處理要好不少,
在屢次計算的時候也能夠很清晰的經過符號看邏輯
總結:對於浮點型的計算,不少人都會採用將金額*100後計算 最後四捨五入爲整型,這種方式看起來也是不會有錯的,在簡單的計算過程當中基本出錯機率忽略不計,可是浮點型的計算偏差會隨着計算量的增大而增大,因此在金額的計算中,仍是採用bc類方法計算爲好。
/** * PHP精確計算 主要用於貨幣的計算用 * @param $n1 第一個數 * @param $symbol 計算符號 + - * / % * @param $n2 第二個數 * @param string $scale 精度 默認爲小數點後兩位 * @return string */function ncPriceCalculate($n1,$symbol,$n2,$scale = '2'){ $res = ""; switch ($symbol){ case "+"://加法 $res = bcadd($n1,$n2,$scale);break; case "-"://減法 $res = bcsub($n1,$n2,$scale);break; case "*"://乘法 $res = bcmul($n1,$n2,$scale);break; case "/"://除法 $res = bcdiv($n1,$n2,$scale);break; case "%"://求餘、取模 $res = bcmod($n1,$n2,$scale);break; default: $res = "";break; } return $res;}/** * 價格由元轉分 * @param $price 金額 * @return int */function ncPriceYuan2fen($price){ $price = (int) ncPriceCalculate(100,"*", ncPriceFormat($price)); return $price;}/*** 價格格式化** @param int $price* @return string $price_format*/function ncPriceFormat($price) { $price_format = number_format($price,2,'.',''); return $price_format;}