博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
从 Number.isNaN 与 isNaN 的区别说起 例子
阅读量:6328 次
发布时间:2019-06-22

本文共 2676 字,大约阅读时间需要 8 分钟。

例子

大家先看一看下面这个例子,

isNaN(NaN);isNaN('A String');isNaN(undefined);isNaN({});Number.isNaN(NaN);Number.isNaN('A String');Number.isNaN(undefined);Number.isNaN({});

如果你能很清楚答案,那么这篇文章你可以略过。

不清楚的朋友,我们来慢慢来分析。

答案如下:

isNaN(NaN); // trueisNaN('A String'); // trueisNaN(undefined); // trueisNaN({}); // trueNumber.isNaN(NaN); // trueNumber.isNaN('A String'); // falseNumber.isNaN(undefined); // falseNumber.isNaN({}); // false

为什么看起来同样的函数,得出的结果为什么不同呢?

NaN 是什么?

在解释 NaN 之前,我想先解释下 type/value/variable 这个几个容易混淆的概念,已经很清楚的朋友可以跳过这一小节

type / value / variable 是什么

在 JavaScript 中,value一共有七种type

  1. null
  2. undefined
  3. boolean
  4. number
  5. string
  6. object
  7. symbol (ES6新增)

那么,variable是什么呢?就是我们平时 var 之后的声明的那个东西。

type, value, variable 之间的关系可以这么说:variable是存放value的容器,而value是有着type概念的,但是容器variable是没有type的概念的,举个例子

var a = 'foo';

容器 variable a 装着 value 'foo', value 'foo' 的type是string

NaN

MDN里面这么描述

The global NaN property is a value representing Not-A-Number.

意思是是说:NaN是一个放在 global(浏览器里是window)对象里的一个value,是一个代表Not-A-Number的value.

意思还是很含糊。

那么我们在看神书《You Don't Know JS》里的描述

NaN literally stands for "not a number", though this label/description is very poor and misleading, It would be much more accurate to think of NaN as being "invalid number," "failed number," or even "bad number," than to think of it as "not a number."

根据上一个小结的知识,我们知道了,NaN是一个 value, 这个 value 的 type 是 number。

但是跟普通的type是number的value不一样的是,NaN 代表 'Not a number' 这一意义。

那么问题来了,怎么判断一个 value 是不是 NaN 呢?

isNaN()

也许有人会说,判断还不容易吗?直接比较不就好了。

NaN === NaN // false

NaN 跟它自己比较会返回false。

所以,我们就需要一个特殊的函数来判断一个value是不是NaN了。

isNaN() 就横空出世了。

我们再回头看一看上面的例子

isNaN(NaN); // true

OK, 成功了,看似很完美,但是接着看以下例子

isNaN('A String'); // trueisNaN(undefined); // trueisNaN({}); // true

会发现,很明显不是 NaN 的 value 也被误判成 NaN 了。

这个BUG已经存在了20年,从JavaScript最开始就一直存在。很明显当初的设计者,在设计isNaN()的时候,局限了在 "Not a Number" 这一字面意思上了:只要不是number就会返回 true。

于是 ES6 为了弥补这一BUG(而不是修正,因为isNaN存在时间太长,有可能很多功能都是基于这个BUG之上的)引入了 Number.isNaN().

Number.isNaN(NaN); // trueNumber.isNaN('A String'); // falseNumber.isNaN(undefined); // falseNumber.isNaN({}); // false

回头看上面的例子,就明白了修复了什么问题。

Number.isNaN() 的 polyfill

没有ES6的情况下,可以采用以下polyfill

if (!Number.isNaN) {  Number.isNaN = function(n) {    return (      typeof n === "number" &&      window.isNaN( n )    );  };}

简单来看,就是在原有 isNaN() 的基础上增加了一个 type 的判断,因为 NaN 的 type 是 number。

还有一种更加简单的实现

if (!Number.isNaN) {    Number.isNaN = function(n) {        return n !== n;    };}

利用了只有 NaN 不跟自己相等的特性。

顺便吐槽一下MDN的解释,他是这么解释 isNaN()

You could think of isNaN as:

var isNaN = function(value) {    return Number.isNaN(Number(value));}

他是在ES6新函数Number.isNaN()的基础上,去解释旧函数isNaN()的。

不过我们可以通过以上方式来解释判断 isNaN() 为什么会出现

isNaN('A String'); // trueisNaN(undefined); // trueisNaN({}); // true

这样的情况了。

转载地址:http://fnwoa.baihongyu.com/

你可能感兴趣的文章
EMMA 覆盖率工具
查看>>
WPF中获取系统本身自带的控件模板(XAML)
查看>>
Aircrack-ng官方文档翻译[中英对照]---Aireplay-ng
查看>>
cxImage控件使用
查看>>
js返回顶部
查看>>
手机测试用例-时钟测试用例
查看>>
Hamming校验码
查看>>
第六十一课、智能指针类模板
查看>>
LoadRunner 文本检查点使用
查看>>
12/4团队会议4
查看>>
[BZOJ1927]星际竞速(费用流)
查看>>
PowerDesigner添加表注释
查看>>
使用VMware安装Ubuntu虚拟机,创建后开启显示黑屏的解决方法
查看>>
Java数据结构与算法(11) - ch06递归(二分法查找)
查看>>
文件操作
查看>>
this的指向
查看>>
Object类及其常用方法简介
查看>>
怎样在Eclipse中使用debug模式调试程序
查看>>
对查询结果进行分页
查看>>
算法模板——并查集 2(支持快速即时查询本连通块内容,纯原创!)
查看>>