laziness, impatience, and hubris PerlMonks

Re: Using (s)printf()

 on Aug 31, 2001 at 16:05 UTC Need Help??

I tried to use the example given in the text. This is what I get (using perl v5.6.1 built for cygwin):      printf("%.2f", 9.333333);

This prints "9.33". Actually, this is what I expect, but the article says it should be 9.34.

If I try the following:      printf("%.1f", 9.55);

I get 9.6, which is OK, but if I try      printf("%.1f", 43.55);

I get 43.5, which is rounded the wrong way.

Can someone explain this? (Please send me an email (oz1lo001@sneakemail.com)).

Replies are listed 'Best First'.
Re: Re: Using (s)printf()
by Hofmator (Curate) on Aug 31, 2001 at 16:20 UTC

• The first one (9.3333 -> 9.34) is a typo in the original.
• The second and third one are easily explained by the following
```printf("%.15f", 9.55);   # prints 9.550000000000001
printf("%.15f", 43.55);   # prints 43.549999999999997
This means that due to the limitations of the internal representation, the literal 9.55 is in fact 9.550000000000001 which rounds to 9.56. The same goes for 43.55 but there the representation is slightly smaller than 43.55 so a rounding down occurs.

-- Hofmator

Actually, this isn't quite the whole story. Most everyone follows the IEEE convention of "round towards nearest or even."
Examples (rounding all of these to the one's place):

2.51 becomes 3 (this is the 'nearest' rule, which always comes first)
2.49 becomes 2 (again, 'nearest')

However, what happens if you have 2.50 ? Which way do you round it... 'tis no nearer to 2 than to 3. The IEEE standard says if there is a tie, round to the even number.

2.50 becomes 2
3.50 becomes 4

You have to pick up or down... this method is consistent and thus tends to make your errors (statistically) smaller.

Actually, this isn't quite the whole story. Most everyone follows the IEEE convention of "round towards nearest or even."
I'm aware of that - but it wasn't applicable in this case. Btw, (s)printf doesn't seem to get it right:
```printf "%.15f", 0.5;  # prints 0.500000000000000
printf "%.0f",  0.5;  # prints 1 and should print 0
and 0.5 can be represented exactly ...

-- Hofmator

Thank you for your explanation. How do I circumvent this problem? I have a little perl-program that pulls these numbers from a file and it has to round them correctly. Isn't there some way to store a number exactly?

What if you want to chop a number but not round it? printf just *wants* to round, no option to tell it not to :(

I needed something that would chop the extra digits (price calculataions) without rounding.

quick fox was to compare the sprintf result with the original and if it was larger remove 0.01 - yuk!...

I ended up using split and substr... - not pretty either

There must be a better way!

Create A New User
Domain Nodelet?
Node Status?
node history
Node Type: note [id://109360]
help
Chatterbox?
and the web crawler heard nothing...

How do I use this?Last hourOther CB clients
Other Users?
Others rifling through the Monastery: (7)
As of 2024-09-13 12:51 GMT
Sections?
Information?
Find Nodes?
Leftovers?
Voting Booth?
The PerlMonks site front end has:

Results (20 votes). Check out past polls.

Notices?
 • erzuuli ‥ 🛈The London Perl and Raku Workshop takes place on 26th Oct 2024. If your company depends on Perl, please consider sponsoring and/or attending.