简单分析[]==[],[]==![],{}=={}和{}==!{}
当比较运算符 == 两边的数据类型相同的时候会直接比较值是否一致,但如果两边的数据是引用类型的话会不一样,引用类型做比较用的是地址,因为每个引用类型地址不同,所以比较结果是 false 。
首先要知道的是这里 ![] 这个优先级比 == 高,所以会先运算 ![] 再执行比较。
逻辑非会将操作值转换成一个布尔值,然后对其取反,而转换布尔值隐式调用的就是 window 下的 Boolean 转换函数。 Boolean 转换函数转换规则如下:
明白了 Boolean 转型函数的一些规则后,就能知道 ![] 会返回 false 。因为它是对象(数组也是对象),所有先转为 true ,再取反,则为 false 。
逻辑非转换完后我们比较的内容就是 [] == false ,这时候可以清晰得发现比较两边是不同的数据类型。
如果是不同的数据类型,是需要先做数据类型转换,然后再进行比较的。
先看看相等操作符 == 的一些转换规则:
分析 [] == ![] 得出以下思路:
[] == ![] 执行结果为 true ,为何同样为引用类型的 {} == !{} 就为 false ?区别是操作符 == 的第三条转换规则, [] 和 {} 的 toString() 方法不同。
在对象上调用 toString 方法,此时是通过继承 Object.prototype 得到的。此时返回值格式为字符串 [object Constructor] 。 Constructor 指的是构造函数。如果是对象, Constructor = Object , 如果是数组, Constructor = Array 。
在数组上调用 toString 方法,此时是通过继承 Array.prototype 得到的。此时在数组原型上重写了 toSring 方法,覆盖了对象上的 toString 方法,在数组原型上的 toString 方法在不传递参数的情况下,会将数组以逗号的形式转换为字符串,实现与 join() 或者 join(',') 相同。不仅是数组,正则和函数也重写了 toString 。
分析 {} == !{} 得出以下思路: