Pomoc - Szukaj - Użytkownicy - Kalendarz
Pełna wersja: Poważny błąd obliczeń w PHP 5.x
Forum PHP.pl > Forum > PHP
icemanwlkp
  1. $historia = Array('1195.61','-400.00','-200.00','-200.00','-395.61');
  2. $brutto = 0;
  3. foreach($historia as $poz)
  4. {
  5. echo $brutto.' + '.$poz.' = ';
  6. $brutto = $brutto + $poz;
  7. echo $brutto.'<BR>';
  8. }


Wynik :

0 + 1195.61 = 1195.61
1195.61 + -400.00 = 795.61
795.61 + -200.00 = 595.61
595.61 + -200.00 = 395.61
395.61 + -395.61 = -1.13686837722E-13 - co to za brednie ?
icemanwlkp
Wiem co to jest liczba zmiennoprzecinkowa i wiem jak rozwiązać ten problem , żeby wynik wyszedł zero , pytam się tylko dlaczego w ostatnim wierszu wynik nie wynosi zero !

Błąd powstaje w tej lini :
  1. $brutto = $brutto + $poz;


nie działa również jeśli zrobimy tak:
  1. $brutto = (float)$brutto + (float)$poz;


nie działa również jeśli zrobimy tak:
  1. $brutto = floatvar($brutto) + floatvar($poz);


nie działa również jeśli zrobimy tak:
  1. $brutto = $brutto + round($poz,2);


rozwiązanie :
  1. $brutto = round($brutto,2) + $poz;


DLACZEGO TAK questionmark.gifquestionmark.gifquestionmark.gif MUSI BYĆ !
Mephistofeles
Skoro wiesz co to float, to powineneś także wiedzieć, że zapisywanie jej przez komputer nie jest dokładne. Do precyzyjnych obliczeń stosuje się typy stałoprzecinkowe, a w PHP http://pl2.php.net/bcmath,
kalmaceta
Za wikipedią
Cytat
Arytmetyka zmiennoprzecinkowa nie jest łączna. To znaczy, że dla x, y i z mogą zachodzić różności:
Zyx
Jakbyś wiedział, co to jest liczba zmiennoprzecinkowa, a zwłaszcza - jak ona działa, to byś wiedział też, dlaczego tak musi być.

Podpowiedź: wartość 1195.61 nie może być reprezentowana dokładnie w binarnym zapisie, a 395.61 ma dokładną reprezentację. Aby było śmieszniej, już przy pierwszym dodawaniu procesor matematyczny (tak, procesor matematyczny, a nie żadne PHP) musi trochę sobie pomanipulować mantysami i w efekcie wykonuje operację, której skutkiem ubocznym jest dalsza utrata precyzji:

Kod
1.1956099999999999e+3 -4.0000000000000000e+2 = 7.9560999999999990e+2
flashdev
Cytat(icemanwlkp @ 18.12.2010, 12:27:49 ) *
  1. $historia = Array('1195.61','-400.00','-200.00','-200.00','-395.61');
  2. $brutto = 0;
  3. foreach($historia as $poz)
  4. {
  5. echo $brutto.' + '.$poz.' = ';
  6. $brutto = $brutto + $poz;
  7. echo $brutto.'<BR>';
  8. }


Wynik :

0 + 1195.61 = 1195.61
1195.61 + -400.00 = 795.61
795.61 + -200.00 = 595.61
595.61 + -200.00 = 395.61
395.61 + -395.61 = -1.13686837722E-13 - co to za brednie ?


Jak to? Przecież liczy bardzo dobrze smile.gif
Przy takich obliczeniach jakie tu wykonujesz to przecież już **e-3 można uznać za zero, a gdzie tam jeszcze do -13 smile.gif
Przecież i tak przed wyświetleniem wartości wypada ją zaokrąglić do całych gorszy, nieprawdaż?

  1. $historia = Array('1195.61','-400.00','-200.00','-200.00','-395.61');
  2. $brutto = 0;
  3. foreach($historia as $poz)
  4. {
  5. echo $brutto.' + '.$poz.' = ';
  6. $brutto = $brutto + $poz;
  7. echo round($brutto, 2)."\n"; // tutaj poprawilem
  8. }


Edit:
Inne wersje PHP i wszystkie koprocesory też mają taki "błąd", więc naywając tak ten temat wprowadzasz czytelników w błąd winksmiley.jpg
kiler129
W niektórych wypadkach nawet taki kod:
  1. $x = 3.14;
  2. echo ($x*2)/(2*$x)+$x-1-$x;


nie wypluje zera smile.gif
To całkiem normalne zachowanie i też się na tym przejechałem ;]
erix
Niby porównywanie floata przez ==. [;
Quadina
Przeczytałem ten temat i jestem w ciężkim szoku. Jak można nazywać cokolwiek błędem, mając FUNDAMENTALNE braki w wiedzy?


P.S. Po za tym od kiedy piszemy liczby zmiennoprzecinkowe w apostrofach? Chyba mnie coś ominęło...
To jest wersja lo-fi głównej zawartości. Aby zobaczyć pełną wersję z większą zawartością, obrazkami i formatowaniem proszę kliknij tutaj.
Invision Power Board © 2001-2025 Invision Power Services, Inc.