PHP 数字格式化器与货币,无法设置精度

16
我想在使用 Intl 扩展NumberFormatter PHP 类来处理货币时将精度设置为 0。然而,我得到了一些奇怪的结果。这里是代码:

$numberFormatter = new NumberFormatter('en-US', NumberFormatter::CURRENCY);
$numberFormatter->setAttribute(NumberFormatter::FRACTION_DIGITS, 0);

echo $numberFormatter->formatCurrency('45', 'USD');

它输出了$45,这正是我想要的。但是,如果我使用相同的设置将货币更改为EUR
echo $numberFormatter->formatCurrency('45', 'EUR');

它输出€45.00(尽管我明确设置了精度为零)。
更奇怪的是,如果我将语言环境设置为fr-FR,则数字会按预期输出:
$numberFormatter = new NumberFormatter('fr-FR', NumberFormatter::CURRENCY);
$numberFormatter->setAttribute(NumberFormatter::FRACTION_DIGITS, 0);

echo $numberFormatter->formatCurrency('45', 'EUR');

它输出 45 €

这是一个bug吗?


你尝试过使用其他货币吗?还是问题只出现在“EUR”上? - Tchoupi
1
这种情况发生在某些货币上(例如,我们有一些单元测试在俄罗斯货币上失败了)。但它也取决于所设置的语言环境,这就是为什么它非常奇怪的原因。 - Michael Gallego
迈克尔,你用的是什么操作系统和 PHP 版本? - taco
在Windows 7,PHP 5.3.12上也有同样的问题:en-US + USD -> $45,fr-FR + EUR -> 45 €,en-US + EUR -> €45.00。 - Maxence
你是如何解决这个问题的? - goyote
6个回答

2

我仍然在使用 ICU 4.8.1 时遇到了这个问题。 - Attila Szeremi

0

您必须提供正确的区域设置和货币组合。例如,fr-FR / fr-bh / fr-ch 支持 €,这就是在 formatCurrency 函数中必须提供的。


0

此函数使用本地信息将数字格式化为货币,我所使用的格式为'%#10n',表示美元:

/**
 * That it is an implementation of the function money_format for the
 * platforms that do not it bear.
 * The function accepts to same string of format accepts for the
 * original function of the PHP.
 * The function is tested using PHP 5.1.4 in Windows XP
 * and Apache WebServer.
 * 
 * format I am are using is '%#10n' for $;
 * 
*/
public static function money_format( $format, $number )
{
    $regex  = '/%((?:[\^!\-]|\+|\(|\=.)*)([0-9]+)?'.
            '(?:#([0-9]+))?(?:\.([0-9]+))?([in%])/';
    if (setlocale(LC_MONETARY, 0) == 'C') {
        setlocale(LC_MONETARY, '');
    }
    $locale = localeconv();
    preg_match_all($regex, $format, $matches, PREG_SET_ORDER);
    foreach ($matches as $fmatch) {
        $value = floatval($number);
        $flags = array(
                'fillchar'  => preg_match('/\=(.)/', $fmatch[1], $match) ?
                $match[1] : ' ',
                'nogroup'   => preg_match('/\^/', $fmatch[1]) > 0,
                'usesignal' => preg_match('/\+|\(/', $fmatch[1], $match) ?
                $match[0] : '+',
                'nosimbol'  => preg_match('/\!/', $fmatch[1]) > 0,
                'isleft'    => preg_match('/\-/', $fmatch[1]) > 0
        );
        $width      = trim($fmatch[2]) ? (int)$fmatch[2] : 0;
        $left       = trim($fmatch[3]) ? (int)$fmatch[3] : 0;
        $right      = trim($fmatch[4]) ? (int)$fmatch[4] : $locale['int_frac_digits'];
        $conversion = $fmatch[5];

        $positive = true;
        if ($value < 0) {
            $positive = false;
            $value  *= -1;
        }
        $letter = $positive ? 'p' : 'n';

        $prefix = $suffix = $cprefix = $csuffix = $signal = '';

        $signal = $positive ? $locale['positive_sign'] : $locale['negative_sign'];
        switch (true) {
            case $locale["{$letter}_sign_posn"] == 1 && $flags['usesignal'] == '+':
                $prefix = $signal;
                break;
            case $locale["{$letter}_sign_posn"] == 2 && $flags['usesignal'] == '+':
                $suffix = $signal;
                break;
            case $locale["{$letter}_sign_posn"] == 3 && $flags['usesignal'] == '+':
                $cprefix = $signal;
                break;
            case $locale["{$letter}_sign_posn"] == 4 && $flags['usesignal'] == '+':
                $csuffix = $signal;
                break;
            case $flags['usesignal'] == '(':
            case $locale["{$letter}_sign_posn"] == 0:
                $prefix = '(';
                $suffix = ')';
                break;
        }
        if (!$flags['nosimbol']) {
            $currency = $cprefix .
            ($conversion == 'i' ? $locale['int_curr_symbol'] : $locale['currency_symbol']) .
            $csuffix;
        } else {
            $currency = '';
        }
        $space  = $locale["{$letter}_sep_by_space"] ? ' ' : '';

        $value = number_format($value, $right, $locale['mon_decimal_point'],
                $flags['nogroup'] ? '' : $locale['mon_thousands_sep']);
        $value = @explode($locale['mon_decimal_point'], $value);

        $n = strlen($prefix) + strlen($currency) + strlen($value[0]);
        if ($left > 0 && $left > $n) {
            $value[0] = str_repeat($flags['fillchar'], $left - $n) . $value[0];
        }
        $value = implode($locale['mon_decimal_point'], $value);
        if ($locale["{$letter}_cs_precedes"]) {
            $value = $prefix . $currency . $space . $value . $suffix;
        } else {
            $value = $prefix . $value . $space . $currency . $suffix;
        }
        if ($width > 0) {
            $value = str_pad($value, $width, $flags['fillchar'], $flags['isleft'] ?
                    STR_PAD_RIGHT : STR_PAD_LEFT);
        }

        $format = str_replace($fmatch[0], $value, $format);
    }
    return $format;
}

-1

在php ini中设置默认语言环境的值,您可以使用以下代码:

ini_set('intl.default_locale', 'de-DE');

要更改数字格式,请使用以下代码:

setlocale(LC_MONETARY, 'de-DE');


-1
你可以使用以下代码对钱数进行格式化:-
$number = 1234.56;
setlocale(LC_MONETARY,"en_US"); // Sets the Default for money
echo money_format("%i", $number);

它会输出:

USD 1,234.56

这并没有回答问题。 - nobody

-1

我认为您需要使用支持目标货币的区域设置。因此,en_US 没有欧元,而 de_DE 和 fr_FR 都有。因此,fr_FR 支持欧元并不奇怪。

看起来这不是一个错误。


1
嗯...这对我来说也是个bug。这意味着如果区域设置为“fr_FR”,我无法格式化美元货币,因为我们这里没有美元? - Michael Gallego
这被确认为libicu中的一个bug。请参考@LGT的回答。 - nobody

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接