And*_*dre 3 php rounding-error money-format
我正在使用money_format,格式为%.2n,但得到了一个奇怪的结果。
我已经整理了示例代码,所以其他人可以自己测试。
<?php
setlocale(LC_MONETARY, 'en_US');
$deal = array(
array(
'amt' => 1350,
'rate' => .75,
'lod' => 47
),
array(
'amt' => 990,
'rate' => .75,
'lod' => 27
),
array(
'amt' => 4180,
'rate' => .75,
'lod' => 65
),
array(
'amt' => 2370,
'rate' => .75,
'lod' => 26
)
);
foreach ($deal as $value) {
$fee = (($value['amt'] / 1000) * $value['rate']) * $value['lod'];
var_dump($fee);
echo '<br />money_format: ', money_format('%.2n', $fee), '<br />number_format: ', number_format($fee, 2), '<br /><br />';
}
Run Code Online (Sandbox Code Playgroud)
输出:
float(47.5875)
money_format: $47.59
number_format: 47.59
float(20.0475)
money_format: $20.05
number_format: 20.05
float(203.775)
money_format: $203.77
number_format: 203.78
float(46.215)
money_format: $46.22
number_format: 46.22
Run Code Online (Sandbox Code Playgroud)
您会注意到第三个结果,203.775 显示为 203.77 美元,而不是 203.78 美元。
我对money_format的理解是否遗漏了什么?
phpfiddle 链接:http ://phpfiddle.io/fiddle/1909516587
这是一个常见的问题 floating point math operations
造成混淆的原因是默认情况下 PHP 中的精度设置得太低,并且在从var_dump.
来源:http : //php.net/manual/en/language.types.float.php
要查看实际数字,您需要设置更高的精度或使用 定义精度printf,这将显示更多来自 的浮点值$fee。
ini_set('precision',32);
var_dump($fee);
//or
printf("%01.32f", $fee);
Run Code Online (Sandbox Code Playgroud)
float(47.587500000000005684341886080801)
money_format: 47.59
number_format: 47.59
float(20.04749999999999943156581139192)
money_format: 20.05
number_format: 20.05
float(203.77499999999997726263245567679)
money_format: 203.77
number_format: 203.78
float(46.215000000000003410605131648481)
money_format: 46.22
number_format: 46.22
Run Code Online (Sandbox Code Playgroud)
结果我们可以看到这number_format是不正确的四舍五入。当给出相同的十进制值时,我们得到一个四舍五入的数字和一个不四舍五入的数字:https : //3v4l.org/pgKDs
echo number_format(23.77499999999997, 2); //returns 23.78
echo number_format(2.77499999999997, 2); //returns 2.77
Run Code Online (Sandbox Code Playgroud)
作为一般规则,尽量不要在涉及精度的情况下使用浮点数学,并且倾向于使用最低可能的面额(从一分钱的意义上说),或者依赖 bcmath 函数并使用乘数来检索分数。 http://php.net/manual/en/book.bc.php
将计算更改为使用 bcmath 将解决与money_format和的问题number_format:
bcscale(4);
foreach ($deal as $value) {
$fee = (float) bcmul(bcmul(bcdiv($value['amt'], 1000), $value['rate']), $value['lod']);
echo 'money_format: ' . money_format('%.2n', $fee) . PHP_EOL;
echo 'number_format: ' . number_format($fee, 2) . PHP_EOL . PHP_EOL;
}
Run Code Online (Sandbox Code Playgroud)
结果:
string(7) "47.5875"
money_format: 47.59
number_format: 47.59
string(7) "20.0475"
money_format: 20.05
number_format: 20.05
string(8) "203.7750"
money_format: 203.78
number_format: 203.78
string(7) "46.2150"
money_format: 46.22
number_format: 46.22
Run Code Online (Sandbox Code Playgroud)