面试官 :[] == ! [] 为什么返回 true ?

在 JavaScript 学习过程中,往往出现一些与我们常理相违背的结果,比如我们今天的论题[] == ![] ,它的返回的结果竟然是true ,这着实让初学者感到困惑。因为根据直觉,[]![]两者应该是不相等的。面试官恰恰就会抓住这一点,对程序员们进行考察。

而本文将结合官方文档对该题进行深度剖析。

看看官方文档怎么说

关于 ==

关于==,文档这样描述:Annotated ES5 11.9.1

根据官方文档,我总结了以下 == 的隐式转换规则,一些简单的就忽略掉了:

  1. 类型相同的比较:

    (1)如果类型是 Undefined 或 Null,返回 true

    null == null; 
    
    

    (2)如果一个是 +0,另一个是 -0,返回 true

    +0 == -0; 
    
    

    (3)如果类型是对象,二者引用同一个对象,返回 true,反之返回 false

    {} == {}; 
    
    
  2. null 和 undefined 比较: 仅当它们之间的比较时,返回 true

null == undefined; 

  1. NaN 比较: NaN 与任何值比较都返回 false,包括NaN自己。

NaN == NaN;   

  1. 字符串与数字比较: 如果其中一个操作数是字符串,另一个是数字,将字符串转换为数字,然后进行比较。

'5' == 5;   

  1. 布尔值与非布尔值比较: 布尔值无法直接比较,需先将布尔值转换为数字(true 转换为 1false 转换为 0),非布尔值也需要转换成数字再比较。

true == 1;  
false == '123'; 

  1. 对象与原始类型比较: 如果其中一个是对象,另一个是原始类型,将对象通过 ToPrimitive 转换为原始类型,然后进行比较。_(即如果原始类型为字符串,则对象转换成字符串再比较;如果原始类为布尔值,则将布尔值与对象都转换成数字进行比较;如果原始类为数字,则将对象转换成数字进行比较。)_

举个例子:

{} == 1;  

ToPrimitive(obj, Number) ==> Number({})

  1. 如果 obj 是基本类型,直接返回

  2. 否则,调用 valueOf 方法,如果得到原始值,则返回

  3. 否则,调用 toString 方法,如果得到原始值,则返回

  4. 否则,报错

首先{}先被ToPrimitive转换成字符串"[object Object]",就相当于直接判断 "[object Object]" == 1,字符串与数字的比较中,又要将字符串转换成数字,"[object Object]"转换成数字为 NaN,而NaN 与任何值比较都为 false

所以 {} == 1 返回 false

  1. 关系转换图如下:


回到[] == ! []

这里判断[]![],两边都是对象,那怎么比呢,我们发现右边还有一个!,我们知道!的优先级是要大于==的,那么先进行非运算。

关于 !

关于!,文档这样描述:Annotated ES5 11.4.9

上表为各数据类型转换成布尔值的结果。

简而言之,!这个运算符会进行两步操作:

  1. !后面的操作数转换成布尔值

  2. 将这个布尔值取反

于是我们判断![],将[]转换成布尔值,我们在官方文档中就知道了,任何对象转换成布尔值都得到ture,然后在取反,得到![]false

最终判断

原式[] == ![]经过!运算将等式右边转换成了 false。即[] == false

接着,根据==隐式转换规则,等式两边为对象和布尔,那么它们都应该转换成数字进行比较。

[]经过ToPrimitive会被转换成字符串""。再将等号两边的字符串""和布尔值false转换成数字 0 ,得到 0 == 0 。打印得到 true

所以实际在进行[] == ! []判断时,在JS引擎内部,会将这行代码执行成这个样子:

[] == ![]

[] == !true 

[] == false 

'' == false 

0 == 0 

看到这里,恭喜你又拿下一道大厂面试题!

希望这篇文章能够为你提供帮助,如果你还有不懂之处,可以反复阅读,或者在评论区留言,学习是一个循序渐进,敢于试错的过程,我们顶峰相见!

  • 13
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Web面试那些事儿

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值