实例分析JS中的相等性判断===、 ==和Object.is()


Posted in Javascript onNovember 17, 2019

相信刚接触JS的人都会被他的想等性判断给整糊涂,看看下面代码,你能答对几个?

NaN === NaN // false
NaN == NaN // false
Object.is(NaN, NaN) // true
0 == false // true
1 == true // true
Number(null) === 0 // true
null == 0 // false

Javascript提供了三种不同的值比较操作,分别是严格相等、宽松相等、以及Object.is,希望看完下面的内容,能够彻底弄清楚他的判断逻辑。

1. 严格相等 x === y判断逻辑

1、如果x的数据类型和y的数据类型不相同,返回false;

2、如果x是Number类型

  • x是NaN,返回false
  • y是NaN,返回false
  • x的值和y的值相等,返回true
  • x是+0,y是-0,返回true
  • x是-0,y是+0,返回true
  • 否则返回false

3、其他类型参照SameValueNonNumber(x, y)

  • 断言:x,y不是Number类型;
  • 断言: x,y的数据类型相同;
  • x是undefined, y是undefined return true;
  • x是null, y是null,return true;
  • x是字符串类型,当且仅当x,y字符序列完全相同时(长度相同,每个位置上的字符也相同)返回true, 否则返回false;
  • 如果x是布尔类型,当x,y都为true或者都为false时返回true,否则返回false;
  • 如果x是symbol类型,当x,y是相同的symbol值,返回true,否则返回false;
  • 如果x,y是同一个对象值,返回true,否则返回false;
NaN === NaN // false
undefined === undefined // true
null === null // true
undefined === null // false

2. 宽松相等 x == y

  1. 如果x,y的类型相同,返回x===y的结果;
  2. 如果x是null, y是undefined, 返回true;
  3. 如果x是undefined, y是null, 返回true;
  4. 如果x是数值,y是字符串, 返回x == ToNumber(y);
  5. 如果x是字符串,y是数值, 返回ToNumber(x) == y;
  6. 如果x是布尔类型, 返回ToNumber(x)==y 的结果;
  7. 如果y是布尔类型,返回 x==ToNumber(y) 的结果;
  8. 如果x是String或Number或Symbol中的一种并且Type(y)是Object,返回 x==ToPrimitive(y) 的结果
  9. 如果Type(x)是Object并且Type(y)是String或Number或Symbol中的一种,返回 ToPrimitive(x)==y 的结果
  10. 其他返回false
12 == '0xc' // true, 0xc是16进制
12 == '12' // true
12 == '12c' // false, 说明ToNumber转换是用的Number()方法

注意:

Number(null) === 0
但是
null == 0 // false,

实例分析JS中的相等性判断===、 ==和Object.is()

2.1 ToNumber将一个值转换为数值类型

详情参考数值类型转换

  • 如果是boolean类型, true返回1,false返回0;
  • 如果是数值,只是简单的传入返回;
  • 如果是null,返回0
  • 如果是undefined, 返回NaN;
  • 如果是字符串,字符串如果只包含数字,则将其转换成十进制数;如果是有效的浮点格式,将其转换成对应的浮点数值;如果是二进制或十六进制将其转换成对应的十进制数值;
  • 如果是对象,调用对象的valueOf()方法,然后依照前面规则转换,如果valueOf返回值是NaN,则调用toString()方法,再依照前面的规则转换返回的字符串

2.2 ToPrimitive

toPrimitive(A)通过尝试调用 A 的A.toString() 和 A.valueOf() 方法,将参数 A 转换为原始值(Primitive);
JS中原始类型有:Number、String、Boolean、Null、Undefined;

不同类型对象的valueOf()方法的返回值:

对象 返回值
Array 返回数组对象本身。
Boolean 布尔值
Date 存储的时间是从 1970 年 1 月 1 日午夜开始计的毫秒数 UTC
Function 函数本身
Number 数字值
Object 对象本身。这是默认情况, 可以覆盖自定义对象的valueOf方法
String 字符串值

// Array:返回数组对象本身
var array = ["ABC", true, 12, -5];
console.log(array.valueOf() === array); // true

// Date:当前时间距1970年1月1日午夜的毫秒数
var date = new Date(2013, 7, 18, 23, 11, 59, 230);
console.log(date.valueOf()); // 1376838719230

// Number:返回数字值
var num = 15.26540;
console.log(num.valueOf()); // 15.2654

// 布尔:返回布尔值true或false
var bool = true;
console.log(bool.valueOf() === bool); // true

// new一个Boolean对象
var newBool = new Boolean(true);
// valueOf()返回的是true,两者的值相等
console.log(newBool.valueOf() == newBool); // true
// 但是不全等,两者类型不相等,前者是boolean类型,后者是object类型
console.log(newBool.valueOf() === newBool); // false

// Function:返回函数本身
function foo(){}
console.log( foo.valueOf() === foo ); // true
var foo2 = new Function("x", "y", "return x + y;");
console.log( foo2.valueOf() );
/*
ƒ anonymous(x,y
) {
return x + y;
}
*/

// Object:返回对象本身
var obj = {name: "张三", age: 18};
console.log( obj.valueOf() === obj ); // true

// String:返回字符串值
var str = "http://www.xyz.com";
console.log( str.valueOf() === str ); // true

// new一个字符串对象
var str2 = new String("http://www.xyz.com");
// 两者的值相等,但不全等,因为类型不同,前者为string类型,后者为object类型
console.log( str2.valueOf() === str2 ); // false

3.同值相等

同值相等由 Object.is 方法判断:

  • 两个值都是 undefined
  • 两个值都是 null
  • 两个值都是 true 或者都是 false
  • 两个值是由相同个数的字符按照相同的顺序组成的字符串
  • 两个值指向同一个对象
  • 两个值都是数字并且
    • 都是正零 +0,
    • 或者都是负零 -0,
    • 或者都是 NaN
    • 都是除零和 NaN 外的其它同一个数字
Object.is('foo', 'foo');  // true
Object.is(window, window); // true

Object.is('foo', 'bar');  // false
Object.is([], []);   // false

var foo = { a: 1 };
var bar = { a: 1 };
Object.is(foo, foo);   // true
Object.is(foo, bar);   // false

Object.is(null, null);  // true

Object.is(true, 'true')  // false

// 特例
Object.is(0, -0);   // false
Object.is(0, +0);   // true
Object.is(-0, -0);   // true
Object.is(NaN, 0/0);   // true

4.零值相等

与同值相等类似,不过会认为 +0 与 -0 相等。

小结

=== 不做类型转换,当两边的数类型不相同时,直接返回false;当前类型相同且都是数值类型的时候,有一个是NaN,那么结果就是false, 另外 +0 === -0

==运算符,当两边操作数类不相同时会做隐式转换,然后才进行比较,这样的话就会出现 false == 0, ‘' == false 等现象, 但是Object.is不会做这种转换

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对三水点靠木的支持。

Javascript 相关文章推荐
解决Extjs 4 Panel作为Window组件的子组件时出现双重边框问题
Jan 11 Javascript
JS中不为人知的五种声明Number的方式简要概述
Feb 22 Javascript
分享JavaScript获取网页关闭与取消关闭的事件
Dec 13 Javascript
最全面的百度地图JavaScript离线版开发
Sep 10 Javascript
js HTML5多媒体影音播放
Oct 17 Javascript
jQuery延迟执行的实现方法
Dec 21 Javascript
JS数组方法push()、pop()用法实例分析
Jan 18 Javascript
小程序跳转到的H5页面再跳转回跳小程序的方法
Mar 06 Javascript
JS Html转义和反转义(html编码和解码)的实现与使用方法总结
Mar 10 Javascript
Vue的Options用法说明
Aug 14 Javascript
微信小程序自定义tabBar的踩坑实践记录
Nov 06 Javascript
vue的项目如何打包上线
Apr 13 Vue.js
AngularJS动态生成select下拉框的方法实例
Nov 17 #Javascript
24行JavaScript代码实现Redux的方法实例
Nov 17 #Javascript
JavaScript如何处理移动端拍摄图片旋转问题
Nov 16 #Javascript
JS Ajax请求会话过期处理问题解决方法分析
Nov 16 #Javascript
vue中注册自定义的全局js方法
Nov 15 #Javascript
微信sdk实现禁止微信分享(使用原生php实现)
Nov 15 #Javascript
微信JSSDK实现打开摄像头拍照再将相片保存到服务器
Nov 15 #Javascript
You might like
几个优化WordPress中JavaScript加载体验的插件介绍
2015/12/17 PHP
ThinkPHP实现分页功能
2017/04/28 PHP
动手学习无线电
2021/03/10 无线电
Javascript在IE下设置innerHTML时出现未知的运行时错误的解决方法
2011/01/12 Javascript
Javascript在IE和FireFox中的不同表现简析
2012/12/03 Javascript
JS获取节点的兄弟,父级,子级元素的方法
2014/01/09 Javascript
探讨:JavaScript ECAMScript5 新特性之get/set访问器
2016/05/05 Javascript
移动端使用localStorage缓存Js和css文的方法(web开发)
2016/09/20 Javascript
js设置文字颜色的方法示例
2016/12/30 Javascript
原生js封装添加class,删除class的实例
2017/11/06 Javascript
vue2手机APP项目添加开屏广告或者闪屏广告
2017/11/28 Javascript
手动用webpack搭建第一个ReactApp的示例
2018/04/11 Javascript
浅谈Vue render函数在ElementUi中的应用
2018/09/06 Javascript
在mpvue框架中使用Vant WeappUI组件库的注意事项【推进】
2019/06/09 Javascript
微信小程序的授权实现过程解析
2019/08/02 Javascript
jQuery轮播图功能制作方法详解
2019/12/03 jQuery
详解vue beforeEach 死循环问题解决方法
2020/02/25 Javascript
Vue路由的模块自动化与统一加载实现
2020/06/05 Javascript
浅谈python中的占位符
2017/11/09 Python
Python中numpy模块常见用法demo实例小结
2019/03/16 Python
Python面向对象之类和实例用法分析
2019/06/08 Python
Python实现bilibili时间长度查询的示例代码
2020/01/14 Python
Python实现疫情通定时自动填写功能(附代码)
2020/05/27 Python
Pycharm插件(Grep Console)自定义规则输出颜色日志的方法
2020/05/27 Python
Python Dict找出value大于某值或key大于某值的所有项方式
2020/06/05 Python
Python环境配置实现pip加速过程解析
2020/11/27 Python
解决Pycharm 运行后没有输出的问题
2021/02/05 Python
Fossil美国官网:化石手表、手袋、首饰及配饰
2019/02/17 全球购物
电子商务专业实习生自我鉴定
2013/09/24 职场文书
在校生自我鉴定
2014/01/23 职场文书
保安岗位职责
2014/02/21 职场文书
信用社主任竞聘演讲稿
2014/05/23 职场文书
技术负责人任命书
2014/06/05 职场文书
2015年度服装销售工作总结
2015/03/31 职场文书
apache基于端口创建虚拟主机的示例
2021/04/22 Servers
PostgreSQL之连接失败的问题及解决
2023/05/08 PostgreSQL