JavaScript判断两个数组相等的四类方法 今日热搜
在JavaScript中,数组本质上是一种特殊的对象,它的类型值会返回 object
。如果我们需要比较两个数组是否相等,不能像比较基本类型(String、Number、Boolean等)一样,使用 ===
(或 ==
) 来判断,所以如果要比较数组是否相等,需要使用一些特殊方法。关于JS类型的判断,可见博文typeof详解。
本文总结了几种数组比较的方法,方便我们碰到类似问题时能快速处理。
(资料图)
循环比较toString方法join方法JSON.stringify需要说明的,这里只针对数组元素为原始类型(String、Number、Boolean)的情况。如果数据元素是复杂类型,如Object、Function等等,则无法通过这些方法进行简单比较了,需要另行特殊处理。而且当存在这类复杂类型时,比较数组是否相等已经意义不大,本文不做讨论。
这也是关于数组的第四篇博文,前面三篇如下:
一文搞懂JavaScript数组的特性一文搞懂前端的所有类数组类型
循环比较使用循环的方法一一比较数组元素的值,可能是我们最先能想到的方式。循环比较大概可分为两种方式,一种是使用 for
、while
等循环语句,相对简单,如下代码所示:
const arr1 = [1, 2, 3], arr2 = [1, 2, 3]// 需要判断两个数组长度arr1.length === arr2.length// 一一比较元素值,有一个不相等就不等for (let i = 0; i < arr1.length; i++) { if (arr2[i] !== arr2[i]) { return false }}
另外一种就是使用数组的循环类实例方法,如 foreach
、map
等处理数组循环的实例方法,和使用 for
语句较类似,同样能达到目的。而使用 every
、some
、filter
等这类实例方法,则代码实现上会更简单一些,如下所示:
const arr1 = [1, 2, 3], arr2 = [1, 2, 3]// 使用everyarr1.length === arr2.length && arr1.every((v,i) => v === arr2[i])// 使用somearr1.length === arr2.length && !arr1.some((v, i) => v !== arr2[i])// 使用filterarr1.length === arr2.length && arr1.filter((v, i) => v !== arr2[i]).length === 0// 使用find和findIndexarr1.length === arr2.length && arr1.findIndex((v, i) => v !== arr2[i]) === -1
toString当我们进行循环相关的比较的时候,都使用的是严格相等
===
,会先判断类型是否相等。但如果需要忽略元素类型时,可以使用==
,这样,会自动对数组元素进行类型转换后再比较,如true == 1
会成立。
toString
方法是Object类型对象的实例方法,而JS中Object是几乎所有类型的基类,所以其他类型都能调用该方法。(null和undefined例外,没有实例方法。)toString方法的作用是返回一个对象的字符串形式,这里,我们用它来返回数组的字符串形式的数据。
[1,2,3].toString() // "1,2,3"
以上代码,就是返回数组 [1,2,3]
的字符串形式,以逗号分给元素组成字符串数据,返回的 "1,2,3"
。如果数组元素是复杂类型,如Object对象,则toString返回的结果将不同:
[1,2,{}].toString() // "1,2,[object Object]"
这里toString方法对 {}
直接返回的是 [object Object]
。
鉴于此,我们比较数组元素为原始类型的数组时,可以如下这样使用:
[1,2,3].toString() === [1, 2, 3].toString() // true
需要注意的是,如果数组元素为数字的字符串形式,结果也是相同的:
["1", 2, 3].toString() // "1,2,3"[1,2,3].toString() === ["1", 2, 3].toString() // true
joinjoin
方法是数组的一个实例方法。它有一个可选参数,可以作为分隔符,以该分隔符分隔所有数组元素组成字符串数据返回,如果不加参数,默认是以逗号分割。通过join方法的用法,我们就能知道,如果不提供分隔符的参数,它对数组所起的作用看上去和 toString
方法几乎一样。
[1,2,3].join() === [1, 2, 3].join() // true[1,2,3].join() === ["1", "2", "3"].join() //true
当然,也可以添加分隔符,效果是一样的,如使用空字符:
[1,2,3].join("") // "123"[1,2,3].join("") === ["1", "2", "3"].join("") //true
由上可知,既然join不带参数和toString方法几乎一样,那它们互相之间的比较,本质上都是数组转换成逗号分隔的字符串,所以也是相等的:
[1, "2", true].join() === [1, "2", true].toString() // true[1, "2", true].toString() === [1, "2", true].join() // true
JSON.stringifyJSON.stringify
用于将一个对象或值转换成JSON字符串,如果是数组,一般会这样转换:
JSON.stringify([1, "2", true]) // "[1,"2",true]"
利用这个特点,我们就能通过它进行数组的常规比较:
[1, 2, 3] === [1, 2, 3] // falseJSON.stringify([1, 2, 3]) === JSON.stringify([1, 2, 3]) // trueJSON.stringify([1, "2", true]) === JSON.stringify([1, "2", true]) // true
需要注意的是,JSON.stringify处理字符串是转换结果会带双引号:
JSON.stringify(["1"]) // "["1"]"JSON.stringify(["1"]) === "["1"]" // trueJSON.stringify(["1"]) === "[\"1\"]" // false
关于JSON.stringify方法更多的知识,可查看博文JSON方法详解
当数组元素是空元素、null、undefined时以上介绍的数组元素的类型都是原始类型(String、Number、Boolean),但数组元素还可以是另外三种特殊情况:空元素
、null
、undefined
,接下来将简单介绍下出现这三种情况时的比较方式。
当使用 JSON.stringify
方法时,空元素、null、undefined这三种类型的元素都会被转换成 null
字符串值,可以很好的判断:
JSON.stringify([1, "2", true, , null, undefined]) // "[1,"2",true,null,null,null]"JSON.stringify([1, "2", true, , null, undefined]) === JSON.stringify([1, "2", true, , null, undefined]) // true
toString
和 join
方法较类似,他们会把这三种类型的值都转换成空字符:
[1, "2", true, , null, undefined].join() // "1,2,true,,,"[1, "2", true, , null, undefined].toString() // "1,2,true,,,"[1, "2", true, , null, undefined].toString() === [1, "2", true, , null, undefined].join() // true
当我们使用上面介绍的第一种循环数组元素的方法进行比较时,null和undefined只需要注意它们两种类型的值是否相等:
null == undefined // truenull === undefined // false
使用严格相等比较的时候,这两种类型不相等。
空元素数组元素是空元素时,使用循环方式处理,则会有一些不一样,主要和数组的空元素的特点有关:
数组通过下标读取空元素时,返回undefined。当使用for
、while
、for-of
、find
、findIndex
等语法时,空元素会返回 undefined
值;undefined值能被比较,所以这时候进行数组元素比较时,结果是正确的。当使用数组实例方法循环如 forEach
、map
、every
、some
、filter
等方法时,空元素会被跳过;由于值被跳过,在使用这些方式进行数组元素比较时,结果可能是错误的。const arr1 = [1, 2, , 3], arr2 = [1, 2, 2, 3]arr1.length === arr2.length && arr1.every((v, i) => v === arr2[i]) // truearr1.length === arr2.length && arr1.filter((v, i) => v !== arr2[i]).length === 0 // truearr1.length === arr2.length && !!arr1.find((v, i) => v !== arr2[i]) // falsearr1.length === arr2.length && arr1.findIndex((v, i) => v !== arr2[i]) === -1 // false
以上代码,数组arr1有一个空元素,与数组arr2的元素并不相同,但是我们使用 every
和 filter
比较得到的值为 true
,这显然是不对的;而使用 find
和 findIndex
比较结果为 false
,是正确的。
以上四类数组比较的方式,依据我测试的结果,速度上,循环方式中的 for
语法是最快的,而且该方式还能正确比较空元素、null、undefined三种特殊情况,综合上看使用for循环是最佳选择。
第一种循环类方式,性能表现整体优于另外三种方式。join方法比toString方法更耗时。如果数据量不是很大,这几种方式耗时可能都在0.1ms以内,几乎可以忽略。
最后,我们总结下以上内容,主要介绍了四种能够比较常规数组(数组为原始数据类型)是否相等的四种方式。也介绍了当数组元素是另外三种特殊情况(空元素、null、undefined)时,上面介绍的四种比较方式是否有效。当然,如果数组元素是复杂数据类型如Object、Function等则比较无意义,不在本文讨论范围内。
关键词:
责任编辑:meirong
-
JavaScript判断两个数组相等的四类方法 今日热搜
-
ST龙净披露终止投资华泰保险进展,公司遭立案股民可索赔
-
卫华集团发布"新中式起重机"标准
-
吸血鬼殿下们请小心txt下载_吸血鬼殿下们 请小心
-
环球观速讯丨长恨歌演出完整版_长恨歌演出
-
用益-信托产品快报(2023.06.19):本期业绩比较基准为5.4%-7.4%
-
每日讯息!雷霆s
-
专升本院校查询_专升本院校|天天快资讯
-
江苏泰州可提供海尔洗衣机维修服务地址在哪
-
广东省邮政快递单号查询_广东省邮政速递查询
-
冷水滩区黄阳司镇端午民俗文化活动将于6月21日至22日举办
-
影像继续深卷 vivo X100 Pro+或将使用两亿像素潜望式长焦_环球今热点
-
环球简讯:汉字找茬王整理特等舱怎么通关
-
回收黄金首饰多少钱一克(2023年6月19日)
-
【短讯】【VIP机会日报】三大指数全线回调 AI方向逆势拉升 服务器领域多家公司创历史新高(梳理)
-
全球观热点:中辉期货聚酯日报20230619:TA偏弱运行
-
微速讯:苏州高中转入中等职业学校需满足哪些条件?
-
第七届“创客广东”云浮市中小企业创新创业大赛初赛评审结果公布_环球快资讯
-
天天视点!粽叶飘香迎端午 幸福和谐邻里情
-
天天动态:在岸人民币兑美元收盘报7.1583
-
洛阳市人民政府任免国家工作人员-全球通讯
-
5月份70个大中城市统计显示:新建住宅价格环比上涨城市减少
-
《云顶之弈》S9枪手阵容推荐攻略
-
世界短讯!怎么用身份证查学历_学历验证证明怎么弄
-
上海宝冶承建的山西华鑫源炼铁项目高炉冷却壁开始安装 当前滚动
-
京西大悦城火爆开业,您可曾留意人群里这抹“警”色?-天天头条
-
一公司端午节只发了三颗荔枝 员工吐槽:是不是公司要凉了?
-
北京三里屯中学照片_北京三里屯中学怎么样-天天最新
-
扎波罗热州当地官员:乌军以“巨大损失”占领当地一村庄
-
cettic官网 cettic对外汉语教师资格证
-
公司问答|卫星化学:产业园项目资金来源主要为自有资金 看好该行业前景 环球播报
-
桥西:邻里迎端午 粽香情更浓-环球快看点
-
智己刘涛:疯起来,我连自己都黑
-
6月19日江西地区萤石市场价格暂稳
-
国家卫健委开展老年痴呆防治促进行动 ( 2023-2025 年 ) ,加强老年人健康教育