JavaScript 有两种比较方式:严格比较运算符和转换类型比较运算符。

在相等运算符中对应 ===!====!=

先举个栗子

var str = '1'
var num0 = 0
var num1 = 1
var blT = true
var blF = false
var nul = null
var und = undefined
console.log(str == num1) // true
console.log(str == blT) // true
console.log(blT == num1) // true
console.log(blF == num0) // true
console.log(nul == und) // true console.log(str === num1) // false
console.log(str === blT) // false
console.log(blT === num1) // false
console.log(blF === num0) // false
console.log(nul === und) // false

相等操作符(==)会为两个不同类型的操作数进行类型转换,然后进行严格比较。

严格相等操作符(===),一般也叫做全等操作符。会先判断类型,再比较值是否相等。

类型转换

类型转换的途径:

类型(x) 类型(y) 结果
undefined null true
字符串 数字 toNumber(x) == y
布尔值 数字 toNumber(x) == y
对象 字符串/数字 toPrimary(x) == y

对于两个类型不同的操作数比较。

  • 字符串、布尔值都会先转换成数字类型,再做比较。toNumber(x)
  • 对于复杂数据类型,会先将其转换为原始值,之后,再根据上一个规则比较。

    x = toPrimary(obj) ==> toNumber(x)

    而,如何转成原始类型值呢,一般都会自动通过自带的valueOf()方法或者toString()方法实现。如果转换之后非原始值,比较操作会报类型错误。

具体过程:

先执行valueOf方法,如果该方法返回一个原始值,则将这个原始值转换为数字,并返回这个数字,转换过程结束。如果返回非原始值,继续下一步。

否则,执行toString方法,如果该方法返回一个原始值,则将这个原始值转换为数字,并返回这个数字,转换过程结束。如果返回非原始值 ,继续下一步。

以上都没有成功转换为原始值,则抛出一个类型转换错误的异常。

tips: 每个方法只执行一次。不会将方法返回的非原始类型值继续转换。

举个例子说明下,参考于知乎:

原始数据类型(string,number,boolean,null, undefined,symbol)。

复杂数据类型(Object, Function, Array, Date, ...)

基础数据类型的比较,是值的比较,Object对象的比较,则是引用的比较

相等操作符(==),经过类型转换之后,才比较他们的值或者对象值。

对于全等操作符(===),等号两边的值和类型,必须完全相等。

为什么建议使用 ===

1. review代码时候,增加理解时间

在使用了 ==的表达式中,我们需要先理解作者的意图。

  • 想当然的以为都可以。
  • 确实需要通过类型转换来判断。
  • 不应该类型转换,但是写错了。

2. 会判断错误

let num = 2
let obj = {
valueOf: function() {
return 2
}
}
console.log(obj == 2)

输出true

我们本意是这两者!=,而这里得到的结果是==,这不是我们想要的结果。

3. 会产生异常

let num = 2
let obj = {
valueOf: function() {
return {}
},
toString: function() {
return {}
}
}
console.log(obj === 2)
console.log(obj == 2)

第一行输出:false

第二行输出错误提示:Uncaught TypeError: Cannot convert object to primitive value

通过前面类型转换,可以得知,一般情况下,对象会先转换为原始值。

而其过程是通过自带的valueOf()/toString()先转换为primitive value,再和其他值做比较。

而在这里,我们手动将obj对象的valueOf()/toString()覆盖了,导致无法实现转换为原始值。

一般情况下,建议使用=== ,除非你了解,确实需要类型转换,才使用 ==

条件表达式语句(if、三目运算等)

if (condition) {
statement1
} else {
statement2
}

以上是if语句的基础用法。通过判断condition的布尔值,来决定执行statement1代码块1,还是执行statement2代码块2。

先看下面的例子:

var s = 'str'
var blT = true
var blF = false
if (s) {
console.log('正确')
} else {
console.log('错误')
}
console.log(s == blT)

以上语句,得到的结果是

正确
false

为什么if语句,会执行第一个代码块。而s == blT 却是false呢?

其实,这个就涉及到类型转换的问题了。

输出“正确”

  • if语句的条件表达式,会自动调用Boolean()转换函数,将这个表达式的结果转换为一个布尔值。
  • Boolean('str') = true

输出false

  • 根据在上面类型转换那部分所讲,会先调用Number()转换函数转换为数值,之后再比较。
  • Number('str') = NaN
  • Number(true) = 1

图表举例列出常见对象相等情况

普通相等

完全相等

【完】

javascript中!=、!==、==、===操作符总结的更多相关文章

  1. 坑:JavaScript 中 操作符“==” 和“===” 的区别

    标题:JavaScript 中 操作符"==" 和"===" 的区别 记录一些很坑的区别: 1. '' == '0' // false 0 == '' // t ...

  2. JavaScript中+操作符的特殊性

    在JavaScript中+操作符有两个作用: (1)加法运算 (2)字符串连接 在使用+操作符进行运算时,当+操作符两边都是数值类型的时候,进行加法运算; 当+操作符两边有任意一边是字符串,则进行字符 ...

  3. javascript中的操作符详解1

    好久没有写点什么了,根据博主的技术,仍然写一点javascript新手入门文章,接下来我们一起来探讨javascript的操作符. 一.前言 javascript中有许多操作符,但是许多初学者并不理解 ...

  4. JavaScript中in操作符(for..in)、Object.keys()和Object.getOwnPropertyNames()的区别

    ECMAScript将对象的属性分为两种:数据属性和访问器属性.每一种属性内部都有一些特性,这里我们只关注对象属性的[[Enumerable]]特征,它表示是否通过 for-in 循环返回属性,也可以 ...

  5. Javascript中void操作符

    Javascript中void是一个操作符,该操作符指定要计算一个表达式但是不返回值. void操作符用法格式如下:1.javascript:void (expression)2.javascript ...

  6. javascript 中的new操作符的理解

    new 操作符 在有上面的基础概念的介绍之后,在加上new操作符,我们就能完成传统面向对象的class + new的方式创建对象,在Javascript中,我们将这类方式成为Pseudoclassic ...

  7. javascript中神奇的(+)加操作符

    javascript是一门神奇的语言,这没神奇的语言中有一个神奇的加操作符. 常用的加操作符我们可以用来做: 加法运算,例如:alert(1+2); ==>3 字符串连接,例如:alert(“a ...

  8. [转] JavaScript中in操作符(for..in)、Object.keys()和Object.getOwnPropertyNames()的区别

    ECMAScript将对象的属性分为两种:数据属性和访问器属性.每一种属性内部都有一些特性,这里我们只关注对象属性的[[Enumerable]]特征,它表示是否通过 for-in 循环返回属性,也可以 ...

  9. JavaScript 中的相等操作符 ( 详解 [] == []、[] == ![]、{} == !{} )

    ECMAScript 中的相等操作符由两个等于号 ( == ) 表示,如果两个操作数相等,则返回 true. 相等操作符会先转换操作数(通常称为强制转型),然后比较它们的相等性. 在转换不同的数据类型 ...

  10. 为什么不要在 JavaScript 中使用位操作符?

    如果你的第一门编程语言不是 JavaScript,而是 C++ 或 Java,那么一开始你大概会看不惯 JavaScript 的数字类型.在 JavaScript 中的数字类型是不区分什么 Int,F ...

随机推荐

  1. R12.2. Start and Stop Procedure

    R12.2. Start and Stop Procedure   Leave a comment Individual Components:  Application(Middle Tier) $ ...

  2. apache 负载测试工具 ab

    1.ab工具是apache自带的工具,可以测试服务器的负载能力 2.ab工具的参数 -v:版本 -c:并发数 -n:请求数 -t: 测试所进行的最大秒数 3.例子:ab -c 100 -n 100 - ...

  3. ubuntu下eclipse新建项目没有java project的解决办法

    装好了eclipse之后却发现新建项目没有java project的选项,大致搜索了一下,并没有发现很好的解决方案(大都是让你重新安装什么的),于是开始瞎鼓捣,并且找到了一个方案: 在终端切换到roo ...

  4. Android studio导入工程很卡及下载网络jar很慢问题总结

    AndroidStudio导入项目一直卡在Building gradle project info,实际上是因为你导入的这个项目使用的gradle与你已经拥有的gradle版本不一致,导致需要下载该项 ...

  5. Junit指定测试执行顺序

    原文链接: Test execution order原文日期: 2012年12月06日翻译日期: 2014年7月2日翻译人员: 铁锚说明: Junit4.11版本及以后才支持,建议升级到最新版本.按照 ...

  6. PS 图像调整算法——阈值

    PS里面这个算法,先将图像转成灰度图像,然后根据给定的阈值,大于该阈值的像素赋值为1,小于该阈值的赋值为0. if x>T, x=1; if x<T, x=0; 原图: 效果图:阈值为 1 ...

  7. FFmpeg与VS2010

    编译FFmpeg是一件痛苦的事情,一般都直接使用Zeranoe FFmpeg Builds. 如果使用这个版本,需要注意ffmpeg的帮助里的一段话: To create import librari ...

  8. 史上最全webview详解

    本文来自:http://www.jianshu.com/users/320f9e8f7fc9/latest_articles WebView在现在的项目中使用的频率应该还是非常高的. 我个人总觉得HT ...

  9. mac os 中如何修改顶栏图标的顺序

    很简单哦! 按住 cmd键同时鼠标选中那个图标,直接拖到你想要的位置即可.

  10. ASP.NET MVC不可或缺的部分——DI(IOC)容器及控制器重构的剖析(DI的实现原理)

    IoC框架最本质的东西:反射或者EMIT来实例化对象.然后我们可以加上缓存,或者一些策略来控制对象的生命周期,比如是否是单例对象还是每次都生成一个新的对象. DI实现其实很简单,首先设计类来实现接口, ...