How can I determine if a Cocoa NSNumber
represents NaN (not a number)?
This emerges, for example, when I parse a string that has an invalid (non-numeric) contents.
How can I determine if a Cocoa NSNumber
represents NaN (not a number)?
This emerges, for example, when I parse a string that has an invalid (non-numeric) contents.
So, I found out that the class property [NSDecimalNumber notANumber]
is just for this purpose. In some languages NaN != NaN, but this isn't the case in Cocoa.
As Mike Abdullah says, the natural way to represent a NaN in Cocoa is with nil
, but [NSNumber numberWithDouble:NAN]
does return a valid object. There is no NSNumber
-specific way of detecting this, but the general way, isnan([foo doubleValue])
, works. If you don’t like functions, you can stick it in a category.
nil
is the natural way to represent things like a string with non-numeric constants, as in the OP’s example. –
Sou For decimals, at least:
[[NSDecimalNumber notANumber] isEqualToNumber:myNumber]
To determine if NSNumber is a NaN, convert it to a double and use the C function isnan()
:
NSNumber *validNumber = [NSNumber numberWithDouble: 1.];
NSLog( @"%d", isnan(validNumber.doubleValue) ); // prints "0"
NSNumber *nanNumber = [NSNumber numberWithDouble: 0./0.];
NSLog( @"%d", isnan(nanNumber.doubleValue) ); // prints "1"
However, you should be careful, because there are other special values, for example:
NSNumber *posInfinity = [NSNumber numberWithDouble: 1./0.];
NSLog( @"%d", isnan(posInfinity.doubleValue) ); // prints "0"
If you want to check for these values as well, it's better to use isnormal()
instead:
NSLog( @"%d", isnormal(validNumber.doubleValue) ); // prints "1"
NSLog( @"%d", isnormal(nanNumber.doubleValue) ); // prints "0"
NSLog( @"%d", isnormal(posInfinity.doubleValue) ); // prints "0"
isnormal(0)
returns 0 (same as isnormal(NAN)
), so if zero is a potentially valid value, check both inan
and isinf
instead. isfinite()
is a better option. –
Selfevident I found that this works, but is it legal?
NSNumber *NaN = [NSDecimalNumber notANumber];
NSDecimalNumber *x = ... fill it somehow with NaN content ...
if ( x == NaN ) ... this works
is NaN
guaranteeed to be a singleton
constant value? Would be cool, but I suppose it is not, since all examples I found use the isEqual
methods.
there is also the function isnan() i found it today.
We can also use a #define defined in math.h as follws
if(isnan(myNumber))
{ // myNumber is NaN .
}
Any Boolean expression with NaN will always return false. But how is that useful?
I was getting Nan back from locationInView: while handling some gestures in an iPhone app. And was very pleased to find that any Boolean expression with NaN will always return false. I put this to use liek below:
//I was using the UIPanGestureRecognizer
, and seems that on TouchUp I would get Nan for the /location .x - .y - pretty reasonible since in this case the end of a touch doesnt have a location.
CGPoint location = [gestureRecognizer locationInView:self];
if ( location.x != location.x || location.y != location.y )
{
return;
}
So then as long .x and .y or legitimate float values, of course they will never be not equal to there own value. BUT in the case of .x or .y being NaN, the comparison will be false. And I can safely avoid computations with Nan.
There isn't really such an object for NSNumber since if it's not a number, then, well, it's not an NSNumber. It's more usual to use a nil object to represent this.
© 2022 - 2024 — McMap. All rights reserved.
[NSNumber isEqualToNumber:[NSDecimalNumber notANumber]]
. This returns YES if the NSNumber is NaN. – Dissentient