为什么null==false返回false,而null>=false返回true?JavaScript比较规则

写作背景

因为不理解为什么null==false返回false,而null>=false返回true,所以对 JavaScript的比较运算符==、===、>=、!=的比较规则 部分的知识进行了查漏补缺。

1. == & ===

==是宽松相等,===是严格相等。两者的区别在于:当两边变量类型不相等时,== 会进行隐式转换。(null、undefined除外)

== 的比较规则

  1. 当两边类型相同时,若是原始类型则比较值,若是对象类型,则比较引用是否一致。
  2. 当两边类型不同时,触发隐式转换。

隐式转换规则

什么时候会触发隐式规则?当比较运算符两边的类型不相等时。(null、undefined除外)

  1. boolean vs any,布尔类型-> number
  2. Number vs String,String -> number
  3. 对象类型 vs 原始类型,对象类型 toPrimitive
  4. (null、undefined) vs any,null与非(null、undefined)的值比较总是返回false。

隐式转换的方法共有四种:toString、toNumber、toBoolean、toPrimitive

对于toString和toNumber:

toString toNumber
null "null" 0 (null可理解为no object)
undefined "undefined" NaN
true/false "true"/""false 1/0
10 "10" 10
"hello"/"123" "hello"/"123" NaN/123
[1,2,3,4]/[]/[1,null,2,undefined,3] "1,2,3,4"/""/"1,,2,,3" "1,2,3,4"/""/"1,,2,,3"
{} [objet Object] [object Object]

对于toBoolean: JS中假值只有6种:false、null、undefined、0、NaN、空字符,其他值转为布尔值全为true。

对于toPrimitive: 获取对象类型的valueOf方法的返回值。如果valueOf不存在/返回的不是原始类型的值,则获取对象的toString()方法的返回值,即遵循ToString规则。如果valueOf和toString都没返回值,则抛出异常。

注意:数组调用valueOf返回数组本身,所以数组一旦toPrimitive相当于调用toString。

2. >=

>= 的比较规则

  • 如果两个变量都是String,则按照字典顺序和Unicode比较.如"cat">"Cat","cat1">"cat","2">"11"
  • 如果两个变量都是原始类型,则转成数值比较。
  • 如果变量是对象类型,则转成原始类型比较(toPrimitive)。

所以null==false返回false,但是null>=false,这是因为 == 比较依据是null与非(null、undefined)都不宽松相等,而 >= 将两边都转成number类型。

3. !=

其算法是先求宽松相等运算符的结果,然后返回相反值。

4. 验收学习成果

js 复制代码
  0 == '' // true
  1 == '1' // true
  1e21 == '1e21' // true
  Infinity == 'Infinity' // true
  true == '1' // true
  false == '0' // true
  false == '' // true
  
  // NaN和任何人都不相等,包括它自己
  
  null == false // false
  undefined == false // false

  [] == ![] // true

  [] == 0 // true
  
  [2] == 2 // true

  ['0'] == false // true

  '0' == false // true

  [] == false // true

  [null] == 0 // true

  null == 0 // false

  [null] == false // true

  null == false // false

  [undefined] == false // true
  // 首先变成 [undefined]==0
  // 接着[undefined].valueOf()返回自身
  // 接着调用toString返回""
  // 所以""==0 返回true

  undefined == false // false
//  undefined 和 非(null、undefined)都返回false
    
    [2,3,4]>=[2,3,4] // true
    [2,3,4]==[2,3,4] // false
    let a = [2,3,4]
    a==a // true
    a==[2,3,4]// false
    [2,3,4]>=[1,2,3] // true
    [2,3,4]>=[1,2,3,999] // true
JS 复制代码
  Number([]) // 0
  Number(['10']) //10

  const obj1 = {
    valueOf () {
      return 100
    },
    toString () {
      return 101
    }
  }
  Number(obj1) // 100

  const obj2 = {
    toString () {
      return 102
    }
  }
  Number(obj2) // 102

  const obj3 = {
    toString () {
      return {}
    }
  }
  Number(obj3) // TypeError
Reference

javascript学习之比较运算符_javascript 连续比较 <-CSDN博客 从一道面试题说起---js隐式转换踩坑合集 - 掘金 (juejin.cn)

相关推荐
苦瓜小生12 分钟前
【前端】|【js手撕】经典高频面试题:手写实现function.call、apply、bind
java·前端·javascript
Wilber的技术分享16 分钟前
【LeetCode高频手撕题 2】面试中常见的手撕算法题(小红书)
笔记·算法·leetcode·面试
和沐阳学逆向35 分钟前
我现在怎么用 CC Switch 管中转站,顺手拿 Codex 举个例子
开发语言·javascript·ecmascript
软件测试媛1 小时前
软件测试常见的面试题(46道)
功能测试·面试·职场和发展
kgduu3 小时前
js之客户端存储
javascript·数据库·oracle
四千岁3 小时前
2026 最新版:WSL + Ubuntu 全栈开发环境,一篇搞定!
javascript·node.js
竹林8183 小时前
从“连接失败”到丝滑登录:我用 ethers.js 连接 MetaMask 的完整踩坑实录
前端·javascript
东离与糖宝3 小时前
金三银四Java校招面经:从双非到大厂Offer,我只准备了这些
java·面试
铭毅天下3 小时前
EasySearch Rules 规则语法速查手册
开发语言·前端·javascript·ecmascript
bjzhang754 小时前
使用 HTML + JavaScript 实现 SQL 智能补全功能
javascript·html·sql智能补全