前言
这两个表达式涉及到 JS 中类型转换和比较操作符的行为。JS 是一种灵活的动态语言,它有一套自己的类型转换规则,有时会导致一些看似奇怪的结果。在理解这两个表达式的行为时,需要深入了解 JS 的类型转换机制以及比较操作符的工作方式。
首先介绍一下 JS 中的类型转换机制,JS 中的类型转换通常发生在两种情况下:隐式类型转换 和 显式类型转换 。隐式类型转换是 JS 自动执行的类型转换,通常发生在需要不同类型的值的运算或比较时。而显式类型转换是通过一些内置函数(如parseInt()
、parseFloat()
、String()
、Number()
等)来显式地进行类型转换。
== 的隐式类型转换
特例:
- null == undefined 为 true;
- NaN不与任何值相等,包括它自己;
- null和undefined不会进行数据转换;
1.数值和非数值类型比较
- 数值和字符串
ini
console.log('Ywis' == 1); // false
console.log('' == 0); // true
console.log('1' == 1); // true
console.log('000' == 0); // true
当一个数据类型是数值,另一个是字符串,会将字符串转换为数值,再与之比较。比如其中'Ywis'
会转换为数字NaN
,所以返回false
。
- 数值和布尔值
arduino
console.log(true == 1); // true
console.log(true == 2); // false
console.log(false == 0); // true
console.log(false == 1); // false
很容易就能看出,这样的比较,会将true
转化成数值1
,false
转化为0
。
- 数值和数组
arduino
console.log([] == 1); // false
console.log([] == 0); // true
console.log(['Ywis'] == 1); // false
console.log([1,2] == 2); // false
在这里,数组会先通过调用toString()
转换为字符串后再转化为数值。比如['Ywis']
先转化为字符串'Ywis'
后再转化为数值NaN
,所以返回false
。
拓展:数组、对象和函数在与其他基本数据类型进行比较时都会先转换为字符串,然后再转换为相应的数据类型。
2. 布尔值和非布尔值类型比较
arduino
console.log(true == 1); // true
console.log(true == '00001'); // true
console.log(false == []); // true
console.log(true == ['Ywis']); // false
当布尔值和非布尔类型比较,会将true
转换为 1,false
转换为 0。将非布尔值类型统统转化成数值,这里的类型转换规则和上面提到的相同。
3. 对象和原始值的比较
arduino
console.log({} == 0); // false
console.log([] == 1); // false
当对象与原始值进行比较时,JS 会先尝试通过调用对象的valueOf()
方法来获取原始值。如果valueOf()
方法返回的是一个原始值,JS 会将其转换成与待比较的原始值相同的类型,然后进行比较。如果valueOf()
方法返回的还是一个对象,则会继续调用对象的toString()
方法,将其返回值转换为原始值,然后再进行比较。
比如{} == 0
,对于空对象{}
它的valueOf()
方法返回的是对象本身,因此 JS 会继续调用 toString()
方法,返回得到一个字符串 "[object Object]"
,再转换为数值NaN
,所以返回false
。
逻辑非 ---!
在操作非布尔值类型时,会进行两步操作:
- 将该数据类型先转换成布尔值类型。
- 再进行取反操作。
arduino
console.log(!""); // true
console.log(![]); // false
console.log(!{}); // false
console.log(!(-4)); // false
注意:
false
:布尔值 falseundefined
:未定义的值null
:空值0
:数字 0-0
:负零NaN
:非数字值''
:空字符串
除了以上七个值外,其他值(包括对象、数组、函数等)在转换为布尔值时都会返回 true
。这种规则使得 JS 中的大多数值被视为真值(truthy)。
面试题讲解🔥🔥🔥
一、 [ ] == [ ] 为 false
这是很好理解的,在 JS 中对象(包括数组)的比较是基于它们的引用地址,而不是它们的内容。即使两个对象的内容相同,它们的引用地址也不同,所以比较结果是 false
。
二、 [ ] == ![ ] 为 true
过程:
- 因为
!
的优先级高于==
,首先将空数组转换为true
,再取反得false
; - 接着是
[] == false
,这是布尔值和非布尔值类型比较,首先将false
转换为0
,然后调用对象的toString
方法,返回一个空对象''
- 最后空对象转换为数值
0
,即0 == 0
,结果返回true。
总结🌸🌸🌸
在面试中,对于类型转换的理解不仅仅是一个基本的知识点,更是一个展示候选人深度理解 JS 语言本质的机会。希望本文能够帮助你更好的理解[] == ![]
中的类型转换机制,在之后的面试之中得心应手,祝你登高望远,心向彼岸。