Pair of concepts may confuse you in Javascript — Number.isNaN() and isNaN()

Truong Phan
ITNEXT
Published in
3 min readMay 19, 2021

--

With a long history of development (and chaos), JS has some messy legacy that can’t be removed to keep the consistency but only improved by new features/functions. This leads to confusion for developers. The series has been written as notes for myself and others to comprehend these concepts and avoid bugs in development.

What is NaN ?

NaNis shorthand for Not A Number, specified in The IEEE Standard for Floating-Point Arithmetic (IEEE 754–2008) for floating-point arithmetic established in 1985. In Javascript context, it is a “property of the global object. In other words, it is a variable in the global scope.”. It has the below characteristics:

* It is considered a Number type
* Equalivent to Number.NaN
* NaNis the only value in JavaScript which is not equal to itself.
* It’s falsy

Take a look at some examples, and think about the results !?

console.log(NaN === NaN) // falseconsole.log(NaN == NaN) // falseconsole.log(NaN !== NaN) // trueconsole.log(NaN != NaN) // trueconsole.log(typeof(NaN)) // numbera = NaN;

Global isNaN()

As you can see NaN even can not be compared to itself, so how we can detect if a variable is NaN, before ES6 we can use the function isNaN() yet considered the following examples.

isNaN(NaN); // trueisNaN('NaN');   // trueisNaN(undefined); // trueisNaN({}); // trueisNaN('Techika.com'); // trueisNaN(''); // falseisNaN('12abcd') // true

To understand this behavior, we need to understand how it works properly.

According to MDN: “When the argument to the isNaN function is not of type Number, the value is first coerced to a Number. The resulting value is then tested to determine whether it is NaN

Then, many people argue that its behavior for non-numeric arguments has been confusing and may cause unexpected results. As the result, the new function was introduced in ECMAScript 2015 (ES6) to solve this problem.

Number.isNaN()

It is a static function from the primitive wrapper object — Number. The most important feature of the function is that it doesn’t force converting the argument to a number. Because NaN is the only value in JavaScript that is not equal to itself, Number.isNaN() has been claimed it is necessary.

Number.isNaN(NaN);        // trueNumber.isNaN(Number.NaN); // trueNumber.isNaN(0 / 0);      // true// e.g. these would have been true with global isNaN()Number.isNaN('NaN');      // falseNumber.isNaN(undefined);  // falseNumber.isNaN({});         // falseNumber.isNaN('Techika.com');   // falseNumber.isNaN(''); // trueNumber.isNaN('12abcd') // false

Conclusion

From my personal point of view, isNaN() may not be a bug as many people thought but it could be considered when you want to focus on the detection of value. The thing is that we need to comprehend its mechanism that it will try to convert arguments to Number. For reliability, we should implement Number.isNaN() when we want to make sure its argument is Number for the comparison.

This post is originally posted from my blog.

--

--