Неформатировать деньги при разборе в PHP - PullRequest
33 голосов
/ 28 февраля 2011

Есть ли способ получить значение с плавающей запятой для такой строки: 75,25 €, отличное от parsefloat(str_replace(',', '.', $var))?

Я хочу, чтобы это зависело от текущего языка сайта, и иногда запятую можно заменить точкой.

Ответы [ 8 ]

45 голосов
/ 04 ноября 2013

Это немного более сложное / медленное решение, но работает со всеми локалями.Поскольку решение @ rlenom работает только с точками в качестве десятичного разделителя, а некоторые локали, например испанский, используют запятую в качестве десятичного разделителя.

<?php

public function getAmount($money)
{
    $cleanString = preg_replace('/([^0-9\.,])/i', '', $money);
    $onlyNumbersString = preg_replace('/([^0-9])/i', '', $money);

    $separatorsCountToBeErased = strlen($cleanString) - strlen($onlyNumbersString) - 1;

    $stringWithCommaOrDot = preg_replace('/([,\.])/', '', $cleanString, $separatorsCountToBeErased);
    $removedThousendSeparator = preg_replace('/(\.|,)(?=[0-9]{3,}$)/', '',  $stringWithCommaOrDot);

    return (float) str_replace(',', '.', $removedThousendSeparator);
}

Тесты:

['1,10 USD', 1.10],
['1 000 000.00', 1000000.0],
['$1 000 000.21', 1000000.21],
['£1.10', 1.10],
['$123 456 789', 123456789.0],
['$123,456,789.12', 123456789.12],
['$123 456 789,12', 123456789.12],
['1.10', 1.1],
[',,,,.10', .1],
['1.000', 1000.0],
['1,000', 1000.0]

Предостережения:Сбой, если десятичная часть имеет более двух цифр.

Это реализация из этой библиотеки: https://github.com/mcuadros/currency-detector

42 голосов
/ 14 сентября 2011

использовать ereg_replace

$string = "$100,000";
$int = ereg_replace("[^0-9]", "", $string); 
echo $int;

выходы

1000000

function toInt($str)
{
    return (int)preg_replace("/\..+$/i", "", preg_replace("/[^0-9\.]/i", "", $str));
}

Обновление


<?php
$string = array("$1,000,000.00","$1 000 000.00","1,000 000.00","$123","$123 456 789","0.15¢");
foreach($string as $s) {
    echo $s . " = " . toInt($s) . "\n"; 
}
function toInt($str)
{
    return preg_replace("/([^0-9\\.])/i", "", $str);
}
?>

Выходы

$1,000,000.00 = 1000000.00
$1 000 000.00 = 1000000.00
1,000 000.00 = 1000000.00
$123 = 123
$123 456 789 = 123456789
0.15¢ = 0.15

и если вы приведете его как целое число

<?php
$string = array("$1,000,000.00","$1 000 000.00","1,000 000.00","$123","$123 456 789","0.15¢");
foreach($string as $s) {
    echo $s . " = " . _toInt($s) . "\n";    
}
function _toInt($str)
{
    return (int)preg_replace("/([^0-9\\.])/i", "", $str);
}
?>

выходы

$1,000,000.00 = 1000000
$1 000 000.00 = 1000000
1,000 000.00 = 1000000
$123 = 123
$123 456 789 = 123456789
0.15¢ = 0

Итак, вот оно.одна строка, одна замена.ты в порядке

42 голосов
/ 28 февраля 2011

Вы можете использовать

Пример из руководства:

$formatter = new NumberFormatter('de_DE', NumberFormatter::CURRENCY);
var_dump($formatter->parseCurrency("75,25 €", $curr));

дает: float(75.25)

Обратите внимание, что расширение intl не включено по умолчанию. Пожалуйста, обратитесь к Инструкции по установке .

9 голосов
/ 14 сентября 2011

Вам нужно удалить символ валюты из строки.PHP intval останавливается на 1-м нечисловом символе, который он находит.

$int = intval(preg_replace('/[^\d\.]/', '', '$100')); // 100

Хотя, если у вас есть значение типа $100.25, вы можете вместо этого использовать floatval.

$float = floatval(preg_replace('/[^\d\.]/', '', '$100.25')); // 100.25
2 голосов
/ 14 сентября 2011

PHP имеет intval ( вот документы ), что (насколько я могу судить) точно так же, как JavaScript parseInt.

Однако, для чегоСтоит, я не думаю, что любая функция поможет вам с тем, что вы пытаетесь сделать.Поскольку первый символ не числовой, оба сходят с ума (PHP даст вам 0, JS даст вам NaN).Таким образом, на любом языке вам придется выполнить синтаксический анализ строк / регулярных выражений.

1 голос
/ 02 октября 2013

Я новичок, так что, вероятно, есть очевидный (для других, не для меня) недостаток в подходе ниже, но я все равно поделюсь им.Мне было бы интересно узнать, работает ли он быстрее или медленнее, чем использование preg_replace, но не проводил никаких тестов скорости.

$badChars = array("$", ",", "(", ")");
$dirtyString = "($3,895.23)";
$cleanString = str_ireplace($badChars, "", $dirtyString);
echo "$dirtyString becomes $cleanString<p>";

$ dirtyString может быть массивом, поэтому:

$badChars = array("$", ",", "(", ")");
$dirtyStrings = array("($3,895.23)", "1,067.04", "$5683.22", "$9834.48");
$cleanStrings = str_ireplace($badChars, "", $dirtyStrings);

echo var_dump($cleanStrings);
1 голос
/ 27 марта 2012

У меня была похожая проблема, когда я не получил символ валюты, только строки (то есть: 1 234 567,89 или 1,234,567,89).

Это помогло мне преобразовать оба случая в числа с плавающей запятой:

$val = str_replace(",", ".", $formatted);
$val = preg_replace("/[\,\.](\d{3})/", "$1", $val);

Но ответ Гордона намного чище.

1 голос
/ 14 сентября 2011

Кастинг - ваш друг:

$int = (int) $string;

Обновление на основе оп :

Попробуйте что-то вроде этого:

<?php

function extract_numbers($string)
{
    return preg_replace("/[^0-9]/", '', $string);
}

echo extract_numbers('$100');

?>

Демо: http://codepad.org/QyrfS7WE

...