Bug in converting float to int?

J

Jeff Vogt

I just tried this on 1.8.6. Is this a bug or am I missing something?
=> 1011

And if this is in fact intended behavior, how can I actually get the
results I want? (Converting dollars and cents into cents)
 
J

Jeff Vogt

Oh and it looks like this only happens with floats ending in .12.

Ruby thinks 12 hundredths is not as big as it's buddies?? :)
 
G

Gary Wright

I just tried this on 1.8.6. Is this a bug or am I missing something?

=> 1011


Ruby floating point values are stored in binary format.
10.12 (decimal) can not be represented exactly in binary:
=> "10.11999999999999921840299066388979554176330566406250"

The default conversion of float to string (e.g., for output) rounds
to 6 (or maybe 7) places and truncates trailing zeros:
=> "10.12"

Float#to_i truncates:
=> 1011
 
C

Clifford Heath

Jeff said:
I just tried this on 1.8.6. Is this a bug or am I missing something?

Wikipedia is good, but I think the best-known article on this is here:
entitled said:
And if this is in fact intended behavior, how can I actually get the
results I want? (Converting dollars and cents into cents)

Avoid using floating point. Seriously. Some might advice you to round:
(a*100).round => 1012, but that's *unreliable*, even with some quite
small number of digits.

Store cents as an integer, or use BigDecimal.

Clifford Heath.
 
J

Jeff Vogt

Gary said:
Ruby floating point values are stored in binary format.
10.12 (decimal) can not be represented exactly in binary:

=> "10.11999999999999921840299066388979554176330566406250"

Clifford said:
Avoid using floating point. Seriously. Some might advice you to round:
(a*100).round => 1012, but that's *unreliable*, even with some quite
small number of digits.

Store cents as an integer, or use BigDecimal.

Thanks everyone, quite interesting info there. It looks like I will be
doing my dollars to cents math a little differently.

Clifford, can you elaborate on why round is unreliable?
 
R

Raphael Clancy

Jeff said:
I just tried this on 1.8.6. Is this a bug or am I missing something?

=> 1011

And if this is in fact intended behavior, how can I actually get the
results I want? (Converting dollars and cents into cents)

Just to tack on my 1.9997438 cents ;)

We had a big go-round about this a while back. This behavior is not
specific to Ruby. The reason Ruby behaves like this is that it adheres
to the IEEE standard for floating point numbers, which is common to most
programming languages. A standard so old that it was first pressed into
clay tablets back when computers were powered by oxen and water wheels.
Ruby keeps this standard because it's fast and it reflects the way that
computers really work with numbers. There are plenty of alternatives for
dealing with floating point numbers which are usually offered as high
precision scientific numerical packages. But these are almost always
slower than the IEEE standard. And, while they are less lousy (I imagine
most could keep two decimal places straight) they are still prone to
precision error, just because computers are, at their very core, integer
only.
 
R

Raphael Clancy

Raphael said:
Just to tack on my 1.9997438 cents ;)

We had a big go-round about this a while back. This behavior is not
specific to Ruby. The reason Ruby behaves like this is that it adheres
to the IEEE standard for floating point numbers, which is common to most
programming languages. A standard so old that it was first pressed into
clay tablets back when computers were powered by oxen and water wheels.
Ruby keeps this standard because it's fast and it reflects the way that
computers really work with numbers. There are plenty of alternatives for
dealing with floating point numbers which are usually offered as high
precision scientific numerical packages. But these are almost always
slower than the IEEE standard. And, while they are less lousy (I imagine
most could keep two decimal places straight) they are still prone to
precision error, just because computers are, at their very core, integer
only.

Just for "fun", I tried the following on g++ 4.01 (PPC version)

#include <iostream>

using namespace std;

int main(int argc, char* argv[]) {
float x = 10.12;
int y = (int) (x * 100.0);
cout << x << " , " << y << endl;
return(0);
}

which generates the familiar output...

10.12 , 1011
 
C

Clifford Heath

Jeff said:
Clifford, can you elaborate on why round is unreliable?

If you've read the Goldberg paper, I doubt I can add anything.

Conversions to/from ASCII are never totally symmetrical.
You can convert to float and back to string and get a
different result, for the simple reason that it's not an
exact one-to-one mapping. For *some* of those patterns,
the change in the last digit will be *propagated* by the
rounding process, resulting in a change to your value.

I recall learning this in 1986 on an HP500. which converted
516.12 to double and back as 516.11 - only five significant
figures.

Clifford Heath.
 

Ask a Question

Want to reply to this thread or ask your own question?

You'll need to choose a username for the site, which only take a couple of moments. After that, you can post your question and our members will help you out.

Ask a Question

Members online

No members online now.

Forum statistics

Threads
474,183
Messages
2,570,965
Members
47,512
Latest member
FinleyNick

Latest Threads

Top