资讯专栏INFORMATION COLUMN

js中抽象相等==

hzx / 2515人阅读

摘要:中抽象相等比较算法大致介绍一下的数据类型的数据类型分为种如果再加上数据类型,一共种与的区别描述一个空值空的对象引用即空指针,被当做一个对象,输出为算是一个吧,输出为。运算符把其值参数转换为非类型对象。

Javascript中抽象相等比较算法
undefined==null
//true
[]==[]
//false
[]==![]
//true
{}==!{}
//false
![]=={}
//false
[]==!{}
//true
[1,2]==![1]
//false
大致介绍一下JS的数据类型

ES5的数据类型分为6种:Undefined Null String Number Boolean Object,如果再加上ES6Symbol数据类型,一共7种;

nullundefined的区别:

null描述一个空值(空的对象引用即空指针),null被当做一个对象,typeOf null输出为"Object"(算是一个bug吧),Number(null)输出为0undefined是预定义的全局变量,表示“缺少值”,typeOf undefined输出为"undefined"Number(undefined)输出为NaN

null是一个关键字,而undefined并不是一个关键字;

原始值概念

js的数据类型其实可以分为两种:原始类型引用类型原始类型又称简单类型基本类型,包括UndefinedNullBooleanNumberString五种。引用类型又称复杂类型,即Object原始类型引用类型分别称为原始值复杂值

简单的说:原始值是固定而简单的值,是存放在栈(stack)中的简单数据段,也就是说,它们的值直接存储在变量访问的位置,原始类型的值被称为原始值

原始类型(primitive type)有以下五种类型:Undefined,Null,Boolean,Number,String

typeOf运算符:

条件 返回值
如果变量是undefined类型 undefined
如果变量是Boolean类型 boolean
如果变量是Number类型 number
如果变量是String类型 string
如果变量是Null类型 object
如果变量是引用类型 object
抽象相等算法
用Type(z)代表z的数据类型,比较运算 x==y,其中x和y是值,产生true或false。
    1.Type(x)与Type(y)相同:
        a.如果Type(x)为Undefined或Null,则返回true
           
        b.如果Type(x)为Number,则:
            i.若x为NaN,返回false
            ii.若y为NaN,返回false
            iii.若x与y数值相等,返回true
            iiii.若x为+0,y为-0,返回true
            iv.若x为-0,y为+0,返回true
            v.返回false
            
        c.如果Type(x)为String,则x和y对应位置的字符完全一样才返回true,否则返回false,
        
        d.如果Type(x)为Boolean,则相同值返回true,否则false
        
        f.当x和y引用同一对象时,返回true,否则,返回false
            
    2.x为undefined,y为null,返回true,反之亦然
            
    3. Type(x)为String,Type(y)为Number,则返回比较ToNumber(x) == y,反之亦然
            
    4.Type(x)为Boolean,则返回比较ToNumber(x)==y的结果,反之亦然

    5.Type(x)为String或Number,Type(y)为Object,则返回比较ToPrimitive(y) == x
    
    6.返回false

再来看看ToBooleanToNumberToPrimitive三个运算符的定义:

ToBoolean
输入类型 结果
Undefined false
Null false
Boolean 不转换
Number 如果参数是-0+0NaN,结果为false,否则为true
String 如果参数是空字符串(长度为零),结果为false,否则为true
Object true
ToNumber
输入类型 结果
Undefined NaN
Null +0
Boolean 参数为true,结果为1,参数为false,结果为+0
Number 不转换
String 参见下文的文法和注释
Object 应用下步骤:1、设原始值为ToPrimitive(输入参数,暗示,数值类型)。2、返回ToNumber(原始值)
ToPrimitive

ToPrimitive运算符接收一个值,和一个可选的期望类型作参数。ToPrimitive运算符把其值参数转换为非类型对象。如果对象有能力被转换为不止一种原语类型,可以使用可选的 期望类型 来暗示那个类型。

输入类型 结果
Undefined 不转换
Null 不转换
Boolean 不转换
Number 不转换
String 不转换
Object 返回该对象的默认值。对象的默认值由期望类型传入作为hint参数调用对象内部方法[DefaultValue]得到

** ToPrimitive这个方法,参照火狐MDN上的文档介绍,大致意思如下:

ToPrimitive(obj,preferredType)

JS引擎内部转换为原始值ToPrimitive(obj,preferredType)函数接受两个参数,第一个obj为被转换的对象,第二个
preferredType为希望转换成的类型(默认为空,接受的值为Number或String)

在执行ToPrimitive(obj,preferredType)时如果第二个参数为空并且obj为Date的实例时,此时preferredType会
被设置为String,其他情况下preferredType都会被设置为Number如果preferredType为Number,ToPrimitive执
行过程如
下:

如果obj为原始值,直接返回;

否则调用 obj.valueOf(),如果执行结果是原始值,返回之;

否则调用 obj.toString(),如果执行结果是原始值,返回之;

否则抛异常。

如果preferredType为String,将上面的第2步和第3步调换,即:

如果obj为原始值,直接返回;

否则调用 obj.toString(),如果执行结果是原始值,返回之;

否则调用 obj.valueOf(),如果执行结果是原始值,返回之;

否则抛异常

Ok,到现在,我们需要了解,toString方法和valueOf方法;

toString用来返回对象的字符串表示。

    let obj = {name:"Tom"}; //"[object Object]"
    let obj = {}; //"[object Object]"
    let arr = [1,2];    //"1,2"
    let arr = [];    //""
    let str = "1";      //"1"
    let num = 1;        //"1"
    let boo = true;     //"true"
    let date = new Date();      //"date Sat Mar 24 2018 00:23:12 GMT+0800 (CST)"
    let nul = null;     //报错
    let und;            //报错

valueOf方法返回对象的原始值,可能是字符串数值bool值等,看具体的对象。

    let obj = {name:"Tom"}; //{name:"Tom"}
    let arr = [1,2];    //[1,2]
    let str = "1";      //"1"
    let num = 1;        //1
    let boo = true;     //true
    let date = new Date();      //1521822331609
    let nul = null;     //报错
    let und;            //报错

简单理解:原始值指的是[Null,Undefined,String,Boolean,Number]五种基本数据类型之一

总结一下==运算的规则:
1. undefined == null,结果是true。且它俩与所有其他值比较的结果都是false。

2. String == Boolean,需要两个操作数同时转为Number。

3. String/Boolean == Number,需要String/Boolean转为Number。

4. Object == Primitive(原始值),需要Object转为Primitive(具体通过valueOf和toString方法)。
栗子
undefined==null
//true
[]==[]
//false
[]==![]
//true
{}==!{}
//false
![]=={}
//false
[]==!{}
//true
[1,2]==![1]
//false
undefined==null

结果为true,不用解释了,记住就行了

[]==[]

先看这两个的类型,typeOf([])得到的是"object"

抽象相等算法1-f,引用同一类型的才算相等

返回false

[]==![]

!取反运算符的优先级高于==,因此先算出![]这个得值,再去使用抽象相等算法进行比较

取反运算符会先调用方法ToBoolean,再去取反

ToBoolean([])返回的是true,因此![]应该为false

[]==![]转换为了[]==false

根据抽象相等算法4条,则我们可以比较[]==ToNumber[false]的值,则可以得到[]==0

再根据抽象相等算法5条,比较ToPrimitive([])==0

由于[]不是Date类型,所以先得到[].valueOf()的值,为[]

再得到[].toString()的值,为""的字符串

以上7,8部可合并为一步即比较[].valueOf().toString(),得到""字符串,此时[]转换为了原始值类型(即五种基本类型中的一种)了。

根据抽象相等算法3,则可以比较ToNumber("")==0,到这里[]==![]转化为了0==0

返回true

{}==![]

表达式右侧,重复上一次的1-5步,可以得到{}==0

根据抽象相等算法5条,ToPrimitive({})==0,得到{}.valueOf().toString()得到一个字符串"[object Object]",是原始类型

根据抽象相等算法3,最后比较ToNumber("[object Object]")==0,转变为1==0

返回false

...

其余的栗子自己算一算吧

结语

自己重新写了写一遍整理了一下思路,如果什么地方没有讲清楚,请指出;

参考:

剖析js中各种蛋疼的类型转换

Javascript中抽象相等比较算法

等...文章

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/93661.html

相关文章

  • js抽象相等

    摘要:字符串和数字之间的相等比较如果是数字,是字符串,则返回的结果如果是字符串,是数字,则返回的结果其他类型和布尔类型之间的相等比较如果是布尔类型,则返回的结果如果是布尔类型。 字符串和数字之间的相等比较 如果Type(x)是数字,Type(y)是字符串,则返回x == ToNumber(y)的结果 如果Type(x)是字符串,Type(y)是数字,则返回ToNumber(x) == y的...

    genedna 评论0 收藏0
  • JS的关系比较与相等比较运算

    摘要:在中的关系比较运算,指的是像这种大小值的关系比较。而相等比较,可区分为标准相等比较与严格相等比较两大种类。 在JS中的关系比较(Relational Comparison)运算,指的是像x < y这种大小值的关系比较。 而相等比较,可区分为标准相等(standard equality)比较x == y与严格相等(strict equality)比较x === y两大种类。严格相等比较会...

    paraller 评论0 收藏0
  • 深入理解JavaScript的类型转换

    摘要:等同于等同于其他类型和布尔类型之间的比较如果是布尔类型,则返回的结果。 showImg(https://segmentfault.com/img/bVburFq?w=796&h=398); 前言 JavaScript作为一门弱类型语言,我们在每天的编写代码过程中,无时无刻不在应用着值类型转换,但是很多时候我们只是在单纯的写,并不曾停下脚步去探寻过值类型转换的内部转换规则,最近通过阅读你...

    W4n9Hu1 评论0 收藏0
  • 深入理解JavaScript的类型转换

    摘要:等同于等同于其他类型和布尔类型之间的比较如果是布尔类型,则返回的结果。 showImg(https://segmentfault.com/img/bVburFq?w=796&h=398); 前言 JavaScript作为一门弱类型语言,我们在每天的编写代码过程中,无时无刻不在应用着值类型转换,但是很多时候我们只是在单纯的写,并不曾停下脚步去探寻过值类型转换的内部转换规则,最近通过阅读你...

    niuxiaowei111 评论0 收藏0
  • 深入理解JavaScript的类型转换

    摘要:等同于等同于其他类型和布尔类型之间的比较如果是布尔类型,则返回的结果。 showImg(https://segmentfault.com/img/bVburFq?w=796&h=398); 前言 JavaScript作为一门弱类型语言,我们在每天的编写代码过程中,无时无刻不在应用着值类型转换,但是很多时候我们只是在单纯的写,并不曾停下脚步去探寻过值类型转换的内部转换规则,最近通过阅读你...

    shuibo 评论0 收藏0

发表评论

0条评论

hzx

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<