使用money_format进行意外舍入,包含示例代码

时间:2014-07-18 16:05:49

标签: 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 />';
    }

输出:

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

您将注意到第三个结果,203.775显示为203.77美元而不是203.78美元。

我对money_format的理解中遗漏了什么?

phpfiddle链接:http://phpfiddle.io/fiddle/1909516587

2 个答案:

答案 0 :(得分:4)

这是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);

精确示例:https://3v4l.org/ue0Th

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

因此,我们可以看到number_format错误地舍入。当给出相同的十进制值时,我们得到一个向上舍入的数字和一个不是:https://3v4l.org/pgKDs

的数字
echo number_format(23.77499999999997, 2); //returns 23.78
echo number_format(2.77499999999997, 2); //returns 2.77

作为一般规则,尽量不要使用涉及精度的浮点数学并且支持最低可能的面额(在货币意义上为一分钱),或者依赖于bcmath函数并使用乘数来检​​索小数值。 http://php.net/manual/en/book.bc.php


更改计算以使用bcmath将解决money_formatnumber_format的问题:

BC Math示例:https://3v4l.org/JEZds

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;
}

<强>结果:

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

答案 1 :(得分:0)

国际支持的解决方案如下:

<?php

setlocale(LC_MONETARY, 'en_US');

function _money_format($number, $decimals=2){
    $number = number_format($number, $decimals);
    $local_settings = localeconv();
    $currency_symbol = $local_settings['currency_symbol'];
    return $currency_symbol . $number;
}

echo _money_format('23.4567');

?>

仍然使用number_format得到货币符号+,这应该得到你需要的东西。

在这里演示:http://phpfiddle.io/fiddle/772115651