为什么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)

相关推荐
她似晚风般温柔7893 小时前
Uniapp + Vue3 + Vite +Uview + Pinia 分商家实现购物车功能(最新附源码保姆级)
开发语言·javascript·uni-app
王中阳Go4 小时前
字节跳动的微服务独家面经
微服务·面试·golang
Jiaberrr4 小时前
前端实战:使用JS和Canvas实现运算图形验证码(uniapp、微信小程序同样可用)
前端·javascript·vue.js·微信小程序·uni-app
everyStudy4 小时前
JS中判断字符串中是否包含指定字符
开发语言·前端·javascript
Ylucius5 小时前
动态语言? 静态语言? ------区别何在?java,js,c,c++,python分给是静态or动态语言?
java·c语言·javascript·c++·python·学习
200不是二百5 小时前
Vuex详解
前端·javascript·vue.js
LvManBa5 小时前
Vue学习记录之三(ref全家桶)
javascript·vue.js·学习
深情废杨杨5 小时前
前端vue-父传子
前端·javascript·vue.js
司篂篂7 小时前
axios二次封装
前端·javascript·vue.js
姚*鸿的博客7 小时前
pinia在vue3中的使用
前端·javascript·vue.js