Usuń bezużyteczne cyfry zerowe z miejsc dziesiętnych w PHP


163

Próbuję znaleźć szybki sposób usunięcia zero decimalsz wartości liczbowych, takich jak ten:

echo cleanNumber('125.00');
// 125

echo cleanNumber('966.70');
// 966.7

echo cleanNumber(844.011);
// 844.011

Czy istnieje jakiś zoptymalizowany sposób, aby to zrobić?


4
Ponieważ wartości są w rzeczywistości łańcuchami, dlaczego nie użyć po prostu rtrim () - php.net/manual/en/function.rtrim.php - z „0”. jako drugi argument
Mark Baker

Przepraszam, nie było do końca jasne, nie zawsze są to struny, rozwiązuję pytanie.
vitto

9
@Mark Baker: to złe podejście, ponieważ numer 120 zostanie przycięty do 12
machineaddict

@machineaddict - byłoby to wtedy i tylko wtedy, gdy wartości łańcuchowe nie miałyby kropki dziesiętnej ..... wszystkie cytowane przez OP mają kropkę dziesiętną.
Mark Baker,

2
Następnie powinieneś dodać do odpowiedzi pogrubioną uwagę, że działa to tylko wtedy, gdy są miejsca po przecinku.
machineaddict,

Odpowiedzi:


354

$num + 0 Zrób sztuczkę.

echo 125.00 + 0; // 125
echo '125.00' + 0; // 125
echo 966.70 + 0; // 966.7

Wewnętrznie jest to równoznaczne z rzucaniem na float z (float)$numlub, floatval($num)ale wydaje mi się to prostsze.


45
Myślę, że to brudna sztuczka. nie wolałbym tego, ponieważ kod nie mówi o jego zachowaniu, tak jak floatval()robi to.
ESCOBAR

4
Zachowanie @ESCOBAR floatvalzwraca wartość zmiennoprzecinkową. Twierdziłbym, że używanie go jako programu formatującego ułamki dziesiętne też nie jest zbyt oczywiste ani czyste.
lafor

2
To jest bardzo niejasne. Nasz zespół właśnie napotkał to w naszym kodzie i przez pewien czas zaintrygowało go trzech programistów. Myślę, że odpowiedź Marka jest lepszą alternatywą. Jest dłuższy, ale jego cel jest jasny.
jfbalanc

2
@jfbalanc//COMMENT
Webinan

1
@ESCOBAR Niezupełnie, to oficjalnie sugerowana metoda typowania ... szukaj "żonglerka typami" na
php.net

101

możesz po prostu użyć floatvalfunkcji

echo floatval('125.00');
// 125

echo floatval('966.70');
// 966.7

echo floatval('844.011');
// 844.011

2
kończy się niepowodzeniem, gdy separatorem tysięcy jest kropka, a znak dziesiętny to przecinek.
Munna Khan

20

Oto czego używam:

function TrimTrailingZeroes($nbr) {
    return strpos($nbr,'.')!==false ? rtrim(rtrim($nbr,'0'),'.') : $nbr;
}

NB Zakłada .się, że jest to separator dziesiętny. Ma tę zaletę, że będzie działać na dowolnie dużych (lub małych) liczbach, ponieważ nie ma rzutowania typu float. Nie zmieni też liczb w notację naukową (np. 1.0E-17).


to nie jest bezpieczne rozwiązanie, a co jeśli cena nie jest dziesiętna? na przykład jeśli programista pomnoży cenę przez liczbę całkowitą, otrzymamy liczbę całkowitą
amin

@alex Ta funkcja działa na liczbach całkowitych, zmiennoprzecinkowych i łańcuchach.
mpen

1
Idealne rozwiązanie jako pomocnik dla bcmath
Kart Av1k

17

Zwykłe dodanie +do zmiennej ciągu spowoduje rzutowanie na (float) i usunięcie zer:

var_dump(+'125.00');     // double(125)
var_dump(+'966.70');     // double(966.7)
var_dump(+'844.011');    // double(844.011)
var_dump(+'844.011asdf');// double(844.011)

13

Dla wszystkich odwiedzających tę stronę, którzy mają ten sam problem z komunikatami, zmień:

$num = number_format($value, 1, ',', '');

do:

$num = str_replace(',0', '', number_format($value, 1, ',', '')); // e.g. 100,0 becomes 100


Jeśli są dwa zera do usunięcia, zmień na:

$num = str_replace(',00', '', number_format($value, 2, ',', '')); // e.g. 100,00 becomes 100

Więcej tutaj: numer PHP: przecinek dziesiętny widoczny tylko w razie potrzeby


10

Jeśli chcesz usunąć cyfry zerowe tuż przed wyświetleniem na stronie lub w szablonie.

Można użyć sprintf () funkcji

sprintf('%g','125.00');
//125

‌‌sprintf('%g','966.70');
//966.7

‌‌‌‌sprintf('%g',844.011);
//844.011

Dzięki, podobała mi się ta odpowiedź. To jest dla mnie bardzo pomocne.
Sumit Kumar Gupta,

9

Powinieneś rzucić swoje liczby jako pływaki, co zrobi to za Ciebie.

$string = "42.422005000000000000000000000000";
echo (float)$string;

Wynik tego będzie tym, czego szukasz.

42.422005


1
Rzutowanie pływające może również uszkodzić twój numer. np . (float)'0.1234567890124567890123456789'staje się 0.12345678901246.
mpen

8
$x = '100.10'; 
$x = preg_replace("/\.?0*$/",'',$x); 
echo $x;

Nie ma nic, czego nie można naprawić prostym wyrażeniem regularnym;)

http://xkcd.com/208/


1
Usuwa znaczące elementy, 0takie jak in "100". Nie 0zdejmuje się również "100.010".
mpen

2
Wszyscy cofnijcie się! $x = preg_replace("/(\.0*$)|((?<=\.\d*)0+$)/",'',$x);
Emanuel S.


2

Z powodu tego pytania jest stary. Po pierwsze, przepraszam za to.

Pytanie dotyczy liczby xxx.xx, ale w przypadku, gdy jest to x, xxx.xxxxx lub różnicowy separator dziesiętny, taki jak xxxx, xxxx, może być trudniej znaleźć i usunąć cyfry zerowe z wartości dziesiętnej.

/**
 * Remove zero digits from decimal value.
 * 
 * @param string|int|float $number The number can be any format, any where use in the world such as 123, 1,234.56, 1234.56789, 12.345,67, -98,765.43
 * @param string The decimal separator. You have to set this parameter to exactly what it is. For example: in Europe it is mostly use "," instead of ".".
 * @return string Return removed zero digits from decimal value.
 */
function removeZeroDigitsFromDecimal($number, $decimal_sep = '.')
{
    $explode_num = explode($decimal_sep, $number);
    if (is_array($explode_num) && isset($explode_num[count($explode_num)-1]) && intval($explode_num[count($explode_num)-1]) === 0) {
        unset($explode_num[count($explode_num)-1]);
        $number = implode($decimal_sep, $explode_num);
    }
    unset($explode_num);
    return (string) $number;
}

A oto kod do testu.

$numbers = [
    1234,// 1234
    -1234,// -1234
    '12,345.67890',// 12,345.67890
    '-12,345,678.901234',// -12,345,678.901234
    '12345.000000',// 12345
    '-12345.000000',// -12345
    '12,345.000000',// 12,345
    '-12,345.000000000',// -12,345
];
foreach ($numbers as $number) {
    var_dump(removeZeroDigitsFromDecimal($number));
}


echo '<hr>'."\n\n\n";


$numbers = [
    1234,// 12324
    -1234,// -1234
    '12.345,67890',// 12.345,67890
    '-12.345.678,901234',// -12.345.678,901234
    '12345,000000',// 12345
    '-12345,000000',// -12345
    '12.345,000000',// 12.345
    '-12.345,000000000',// -12.345
    '-12.345,000000,000',// -12.345,000000 STRANGE!! but also work.
];
foreach ($numbers as $number) {
    var_dump(removeZeroDigitsFromDecimal($number, ','));
}

2

Uważaj, dodając +0.

echo number_format(1500.00, 2,".",",")+0;
//1

Wynik tego to 1.

echo floatval('1,000.00');
// 1

echo floatval('1000.00');
//1000

Odpowiadasz na pytanie z 2013 roku.
VTodorov

3
Nie ma problemu z odpowiedzią na stare pytanie, czy może pomóc faktycznym czytelnikom
Reign.85

1

Dziwne, kiedy dostaję z bazy danych numer typu „float” i jeśli mój numer to np. 10000 kiedy go floatval, staje się 1.

$number = $ad['price_month']; // 1000 from the database with a float type
echo floatval($number);
Result : 1

Przetestowałem wszystkie powyższe rozwiązania, ale nie zadziałały.


1
$str = 15.00;
$str2 = 14.70;
echo rtrim(rtrim(strval($str), "0"), "."); //15
echo rtrim(rtrim(strval($str2), "0"), "."); //14.7

0

Skomplikowany sposób, ale działa:

$num = '125.0100';
$index = $num[strlen($num)-1];
$i = strlen($num)-1;
while($index == '0') {
   if ($num[$i] == '0') {
     $num[$i] = '';
     $i--;
   }

   $index = $num[$i];
}

//remove dot if no numbers exist after dot
$explode = explode('.', $num);
if (isset($explode[1]) && intval($explode[1]) <= 0) {
   $num = intval($explode[0]);
}

echo $num; //125.01

powyższe rozwiązania są optymalnym sposobem, ale jeśli chcesz mieć własne, możesz to wykorzystać. Co robi ten algorytm, zaczyna się na końcu łańcucha i sprawdza, czy jego 0 , jeśli tak, ustawia pusty ciąg, a następnie przechodzi do następnego znaku od tyłu, aż ostatni znak jest > 0



0

To moje małe rozwiązanie ... Można dołączyć do klasy i ustawić zmienne

private $ dsepparator = '.'; // ułamki dziesiętne prywatne $ tsepparator = ','; // tysiąc

Można to ustawić przez konstruktora i zmienić język użytkownika.

class foo
{
    private $dsepparator;
    private $tsepparator;

    function __construct(){
        $langDatas = ['en' => ['dsepparator' => '.', 'tsepparator' => ','], 'de' => ['dsepparator' => ',', 'tsepparator' => '.']];
        $usersLang = 'de'; // set iso code of lang from user
        $this->dsepparator = $langDatas[$usersLang]['dsepparator'];
        $this->tsepparator = $langDatas[$usersLang]['tsepparator'];
    }

    public function numberOmat($amount, $decimals = 2, $hideByZero = false)
    {
        return ( $hideByZero === true AND ($amount-floor($amount)) <= 0 ) ? number_format($amount, 0, $this->dsepparator, $this->tsepparator) : number_format($amount, $decimals, $this->dsepparator, $this->tsepparator);
    }
    /*
     * $bar = new foo();
     * $bar->numberOmat('5.1234', 2, true); // returns: 5,12
     * $bar->numberOmat('5', 2); // returns: 5,00
     * $bar->numberOmat('5.00', 2, true); // returns: 5
     */

}

Więc zbudowałem przykładową klasę z małymi próbkami. Lepszy? =)
Paykoman

0

To jest moje rozwiązanie. Chcę zachować możliwość dodawania separatora tysięcy

    $precision = 5;    
    $number = round($number, $precision);
    $decimals = strlen(substr(strrchr($number, '.'), 1));
    return number_format($number, $precision, '.', ',');

0

To jest prosta funkcja jednowierszowa wykorzystująca rtrim, separator zapisu i przecinek dziesiętny:

function myFormat($num,$dec)
 {
 return rtrim(rtrim(number_format($num,$dec),'0'),'.');
 }

0

Uważam, że to rozwiązanie jest najlepsze:

public function priceFormat(float $price): string
{
    //https://stackoverflow.com/a/14531760/5884988
    $price = $price + 0;
    $split = explode('.', $price);
    return number_format($price, isset($split[1]) ? strlen($split[1]) : 2, ',', '.');
}

0

Proste i dokładne!

function cleanNumber($num){
    $explode = explode('.', $num);
    $count   = strlen(rtrim($explode[1],'0'));
    return bcmul("$num",'1', $count);
}

0

Ostateczne rozwiązanie: jedynym bezpiecznym sposobem jest użycie wyrażenia regularnego:

echo preg_replace("/\.?0+$/", "", 3.0); // 3
echo preg_replace("/\d+\.?\d*(\.?0+)/", "", 3.0); // 3

będzie działać w każdym przypadku


0

Poniższe jest znacznie prostsze

if(floor($num) == $num) {
    echo number_format($num);
} else {
    echo $num;
}

-1
$value = preg_replace('~\.0+$~','',$value);

1
Poprawiłoby to jakość Twojej odpowiedzi, gdybyś podał wyjaśnienie, co robi twój kod i jak rozwiąże pierwotne pytanie.
Nigel Ren

@Nikhil Gyan Zgadzam się, że ta odpowiedź nie jest świetna. Nie krępuj się. Jednak przynajmniej nie uważam, że to jest wystarczająco złe dla flagi bardzo niskiej jakości
Sam Hartman

-2

Ten kod usunie zero po przecinku i zwróci tylko dwie cyfry dziesiętne.

liczba $ = 1200,0000;
str_replace ('. 00', '', number_format ($ numer, 2, '.', ''));

Wynik będzie: 1200

Korzystając z naszej strony potwierdzasz, że przeczytałeś(-aś) i rozumiesz nasze zasady używania plików cookie i zasady ochrony prywatności.
Licensed under cc by-sa 3.0 with attribution required.