在javascript中,null>=0 为真,null==0却为假,null的值详解

1.前言
今天看见朋友们在讨论一个问题,说 null 到底和 0 是不是相等的。
听到这里,自己赶紧去写个 Demo 试一下。
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>MR_LP:3206064928</title> </head> <body> </body> <script type="text/javascript"> console.log(null > 0); // false console.log(null < 0); // false console.log(null >= 0); // true console.log(null <= 0); // true console.log(null == 0); // false console.log(null === 0); // false </script> </html>
什么情况?
为什么 console.log(null <= 0); 和 console.log(null >= 0); 这两条的判断是 true 呢?
2.查阅资料
如果想明确,这个问题具体是怎么回事,那么我们需要重新来回顾一下我们的 ECMAScript Language Specification (HTML version),翻译过来就是ECMAScript语言规范(HTML版本)。
2.1 内部相等性运算算法
首先我们来看一下 ES3 关于 内部相等性运算的算法实现。
11.9.3 The Abstract Equality Comparison Algorithm The comparison x == y, where x and y are values, produces true or false. Such a comparison is performed as follows: 1. If Type(x) is different from Type(y), Go to step 14. 2. If Type(x) is Undefined, return true. 3. If Type(x) is Null, return true. 4. If Type(x) is not Number, go to step 11. 5. If x is NaN, return false. 6. If y is NaN, return false. 7. If x is the same number value as y, return true. 8. If x is +0 and y is -0, return true. 9. If x is -0 and y is +0, return true. 10. Return false. 11. If Type(x) is String, then return true if x and y are exactly the same sequence of characters (same length and same characters in corresponding positions). Otherwise, return false. 12. If Type(x) is Boolean, return true if x and y are both true or both false. Otherwise, return false. 13. Return true if x and y refer to the same object or if they refer to objects joined to each other (see 13.1.2). Otherwise, return false. 14. If x is null and y is undefined, return true. 15. If x is undefined and y is null, return true. 16. If Type(x) is Number and Type(y) is String, return the result of the comparison x == ToNumber(y). 17. If Type(x) is String and Type(y) is Number, return the result of the comparison ToNumber(x)== y. 18. If Type(x) is Boolean, return the result of the comparison ToNumber(x)== y. 19. If Type(y) is Boolean, return the result of the comparison x == ToNumber(y). 20. If Type(x) is either String or Number and Type(y) is Object, return the result of the comparison x == ToPrimitive(y). 21. If Type(x) is Object and Type(y) is either String or Number, return the result of the comparison ToPrimitive(x)== y. 22. Return false.
2.2 内部关系运算算法
接下来我们再来看一下 ES3 关于 内部关系运算的算法实现。
11.8.5 The Abstract Relational Comparison Algorithm The comparison x < y, where x and y are values, produces true, false, or undefined (which indicates that at least one operand is NaN). Such a comparison is performed as follows: 1. Call ToPrimitive(x, hint Number). 2. Call ToPrimitive(y, hint Number). 3. If Type(Result(1)) is String and Type(Result(2)) is String, go to step 16. (Note that this step differs from step 7 in the algorithm for the addition operator **+ * in using *and instead of or.) 4. Call ToNumber(Result(1)). 5. Call ToNumber(Result(2)). 6. If Result(4) is NaN, return undefined. 7. If Result(5) is NaN, return undefined. 8. If Result(4) and Result(5) are the same number value, return false. 9. If Result(4) is +0 and Result(5) is -0, return false. 10. If Result(4) is -0 and Result(5) is +0, return false. 11. If Result(4) is +∞, return false. 12. If Result(5) is +∞, return true. 13. If Result(5) is -∞, return false. 14. If Result(4) is -∞, return true. 15. If the mathematical value of Result(4) is less than the mathematical value of Result(5) — note that these mathematical values are both finite and not both zero — return true. Otherwise, return false. 16. If Result(2) is a prefix of Result(1), return false. (A string value p is a prefix of string value q if q can be the result of concatenating p and some other string*r*. Note that any string is a prefix of itself, because r may be the empty string.) 17. If Result(1) is a prefix of Result(2), return true. 18. Let k be the smallest nonnegative integer such that the character at position k within Result(1) is different from the character at position k within Result(2). (There must be such a k, for neither string is a prefix of the other.) 19. Let m be the integer that is the code point value for the character at position k within Result(1). 20. Let n be the integer that is the code point value for the character at position k within Result(2). 21. If m < n, return true. Otherwise, return false.
2.3 ES3 的 运算符
2.3.1 ES3 的 “>” 运算符:
The Greater-than Operator ( > ) The production RelationalExpression : RelationalExpression > ShiftExpression is evaluated as follows: 1. Evaluate RelationalExpression. 2. Call GetValue(Result(1)). 3. Evaluate ShiftExpression. 4. Call GetValue(Result(3)). 5. Perform the comparison Result(4) < Result(2). 6. If Result(5) is undefined, return false. Otherwise, return Result(5).
2.3.2 ES3 的”>=” 运算符:
The Greater-than-or-equal Operator ( >= ) The production RelationalExpression : RelationalExpression >= ShiftExpression is evaluated as follows: 1. Evaluate RelationalExpression. 2. Call GetValue(Result(1)). 3. Evaluate ShiftExpression. 4. Call GetValue(Result(3)). 5. Perform the comparison Result(2) < Result(4). (see 11.8.5). 6. If Result(5) is true or undefined, return false. Otherwise, return true.
2.3.3 ES3 的 “==” 运算符 :
The Equals Operator ( == ) The production EqualityExpression : EqualityExpression == RelationalExpression is evaluated as follows: 1. Evaluate EqualityExpression. 2. Call GetValue(Result(1)). 3. Evaluate RelationalExpression. 4. Call GetValue(Result(3)). 5. Perform the comparison Result(4) == Result(2). (see 11.9.3). 6. Return Result(5).
3. 根据资料得出的内容
着重看一下,上面特意加粗的地方,我们可以明确下面三件事。
那么我们就可以反过来看这个问题了。
null > 0 // null 尝试转型为number , 则为0 . 所以结果为 false, null >= 0 // null 尝试转为number ,则为0 , 结果为 true. null == 0 // null在设计上,在此处不尝试转型. 所以 结果为false.
这里引用一下 Franky大大的话。
a >= b 运算符只是简单的去对 a < b的结果取反. 我以为这是一个设计上的失误的另一个理由是 undefined,在标准中,被单拎出来.细心的你,也一定发现了这一点. 对于undefined的设计, undefined > 0 , undefined < 0, undefined == 0 的结果是符合设计上,逻辑的一致性的. 而null是被遗漏的东西.直到今天早上.我重新翻阅了ES3,5.相关章节. 才恍然大悟自己没有从根本上理解到这个问题.
虽然前面的例子,我catch到了BE当初的设计思想. 但是从全局的角度来看. 从关系运算符到相等运算符,尤其是相等运算符的设计上. 真的十分混乱不堪. BE在信中提到,他对 == 的现状也很无奈. 甚至用愚蠢这个词来形容自己当初的实现(当然他还提到,当初只是为了在10天内设计出js,并跑过qa的测试用例). 即使如此, 但是他仍然表示 null == 0 这个结果是他想要的.
好吧,到了这里,我也有种无力感. 我认为纵观JavaScript,对关系运算和相等运算的设计.除了混乱,我想不出还有什么词来形容它们更恰当. 这一点从,我们生产环境代码中,大量的类型检查,和防御性代码的的存在,就可以证明这一点.
同时 Franky大大还举了另外一个例子。
function case1(a){
if(a == null){
....
}
}
function case2(a){
if(a == undefined){
...
}
}
// 上面两组完全等价, 这就是一种不明确表述.
// 我们永远不知道代码编写者的目的到底是同时匹配null 和 undefined还是只匹配其中某一个
function case3(a){
if(a === null || a === undefined){
...
}
}
// case3 才是最好的表述. 我们明确知道代码编写者的意图.
// 即使很多人可能认为这个代码很愚蠢. 但我坚定的认为这才是好代码.
最后, 不得不提到,我发出null >= 0 这封信后, Andrea Giammarchi 表示了对我之前看法的支持,他同我最初的看法一样,认为 null >= 0 的结果应该为 false . 并建议在 ES7 中的严格模式中,修改这个结果. 虽然同样遭到 David Bruant 的反对. 好吧为他和我的这个错误看法,默哀一分钟…
4.后记
所以写代码,写规范,都应该明确表述. 即使表述的很罗嗦,但不会引起歧义或怀疑. 这才是一份好的标准.文档,代码. 而避免歧义,和各种混乱不堪的规则,是一门语言最应遵守的设计原则.
最后也希望大家在日常的开发中,能够少遇坑。
# javascript
# null
# null的值详解
# null分析
# js判断为空Null与字符串为空简写方法
# JS中判断null、undefined与NaN的方法
# js判断undefined类型
# undefined
# 的区别详细解析
# JavaScript null和undefined区别分析
# PHP处理Json字符串解码返回NULL的解决方法
# js判断输入字符串是否为空、空格、null的方法总结
# JavaScript Undefined
# Null类型和NaN值区别
# JS基础之undefined与null的区别分析
# 区分JS中的undefined
# ""
# 0和false
# 简单方法判断JavaScript对象为null或者属性为空
# 运算符
# 这个问题
# 才是
# 好吧
# 当初
# 看一下
# 来形容
# 则为
# 就可以
# 是一个
# 设计思想
# 最初
# 我也
# 我想
# 是一种
# 对我
# 是他
# 最好的
# 尤其是
# 不出
相关文章:
建站10G流量真的够用吗?如何应对访问高峰?
如何在新浪SAE免费搭建个人博客?
专业型网站制作公司有哪些,我设计专业的,谁给推荐几个设计师兼职类的网站?
公司网站建设制作费用,想建设一个属于自己的企业网站,该如何去做?
微课制作网站有哪些,微课网怎么进?
如何用腾讯建站主机快速创建免费网站?
如何选择高效可靠的多用户建站源码资源?
南平网站制作公司,2025年南平市事业单位报名时间?
建站中国官网:模板定制+SEO优化+建站流程一站式指南
无锡营销型网站制作公司,无锡网选车牌流程?
如何快速生成橙子建站落地页链接?
网站建设设计制作营销公司南阳,如何策划设计和建设网站?
免费制作海报的网站,哪位做平面的朋友告诉我用什么软件做海报比较好?ps还是cd还是ai这几个软件我都会些我是做网页的?
常州自助建站工具推荐:低成本搭建与模板选择技巧
如何快速生成凡客建站的专业级图册?
深圳网站制作平台,深圳市做网站好的公司有哪些?
开封网站制作公司,网络用语开封是什么意思?
如何在云主机上快速搭建多站点网站?
网站制作公司,橙子建站是合法的吗?
建站主机CVM配置优化、SEO策略与性能提升指南
制作网站建设的公司有哪些,网站建设比较好的公司都有哪些?
在线ppt制作网站有哪些软件,如何把网页的内容做成ppt?
如何在橙子建站上传落地页?操作指南详解
邀请函制作网站有哪些,有没有做年会邀请函的网站啊?在线制作,模板很多的那种?
建站主机服务器选购指南:轻量应用与VPS配置解析
如何快速生成高效建站系统源代码?
建站ABC备案流程中有哪些关键注意事项?
网站制作新手教程,新手建设一个网站需要注意些什么?
如何在阿里云部署织梦网站?
如何选择适配移动端的WAP自助建站平台?
如何注册花生壳免费域名并搭建个人网站?
如何用搬瓦工VPS快速搭建个人网站?
南京做网站制作公司,南京哈发网络有限公司,公司怎么样,做网页美工DIV+CSS待遇怎么样?
大连网站设计制作招聘信息,大连投诉网站有哪些?
建站之星如何优化SEO以实现高效排名?
高防服务器租用如何选择配置与防御等级?
建站主机选哪种环境更利于SEO优化?
如何在阿里云高效完成企业建站全流程?
北京网站制作网页,网站升级改版需要多久?
如何挑选最适合建站的高性能VPS主机?
详解jQuery停止动画——stop()方法的使用
如何快速搭建个人网站并优化SEO?
网站制作公司广州有几家,广州尚艺美发学校网站是多少?
胶州企业网站制作公司,青岛石头网络科技有限公司怎么样?
如何用PHP工具快速搭建高效网站?
建站org新手必看:2024最新搭建流程与模板选择技巧
详解免费开源的DotNet二维码操作组件ThoughtWorks.QRCode(.NET组件介绍之四)
北京的网站制作公司有哪些,哪个视频网站最好?
网站制作中优化长尾关键字挖掘的技巧,建一个视频网站需要多少钱?
历史网站制作软件,华为如何找回被删除的网站?
*请认真填写需求信息,我们会在24小时内与您取得联系。