龙空技术网

JavaScript数组去重花样大赏

前端小玫 74

前言:

眼前兄弟们对“js数组能存多少数据啊”大体比较关注,咱们都需要了解一些“js数组能存多少数据啊”的相关资讯。那么小编也在网摘上收集了一些有关“js数组能存多少数据啊””的相关资讯,希望同学们能喜欢,大家快快来了解一下吧!

在平时的前端开发中,常常会遇到数组去重的场景,比如表单校验重复的数据,有重复数据不让提交,或者对后端返回的数据进行去重展示等,同时它也是一道经典的前端JavaScript基础面试题,本期一起来带大家看看去重的各种花样方法。

一、最朴素的方法去重:双重for循环

一想到去重,第一个就想到了双重for循环,遍历两层比较一下就好了,不过要注意的是,用数组的splice删除重复项后,需要将第二层循环的索引减1,不然会出现数组塌陷的问题。

function unique(arr) {	for (let i = 0; i < arr.length; i++) {		for (let j = i + 1; j < arr.length; j++) {			if (arr[i] === arr[j]) {				arr.splice(j, 1);				j--; // 注意:由于删除了元素,需要将 j 减一,以避免漏掉第j项的检查			}		}	}	return arr;}const arr = [1,1,2,2,3,4,4,5,5,6];console.log(unique(arr));// 输出:[ 1, 2, 3, 4, 5, 6 ]
二、最省时间的去重:obj/Map存储 + for循环

由于对象Object的key值唯一,所以我们可以利用这个特性进行去重。核心思路就是用对象或者Map将遍历过的项作为key存储到对象或者Map中,如果遇到已经存在的key,则说明出现了重复项,忽略此项即可。

function unique(arr) {	const obj = {}; // 这里可以换成Map	const newArr = [];	for (let i = 0; i < arr.length; i++) {		const item = arr[i];		if (!obj[item]) {			obj[item] = 1;			newArr.push(item);		}	}	return newArr;}

这也是典型的空间换时间大法,可以让时间复杂度从O(n²)降为O(n),利用对象存储,达到空间换时间的效果,节约执行时间。

三、最好理解的去重:indexOf + lastIndexOf

JavaScript中提供了indexOf和lastIndexOf两个方法,分别拿目标项从数组的第一项和最后一项开始查找,查到了会返回对应的索引值,如果某一项用indexOf和lastIndexOf找到的索引相同,则证明该项不存在重复项,反之就存在重复项。

这种去重方式应该是最好理解的。不过要注意,这里也使用了splice,需要注意处理防止数组塌陷的问题。

function unique(arr) {	for (let i = 0; i < arr.length; i++) {		if (arr.indexOf(arr[i]) !== arr.lastIndexOf(arr[i])) {			arr.splice(i, 1);			i--; // 注意:由于删除了元素,需要将 i 减一,以避免漏掉第i项的检查		}	}	return arr;}
四、最简单的去重:展开运算符 + Set

利用ES6的新特性,也就是展开运算符 + set应该是最简单的去重方式,写的字符数也是最少的。

function unique(arr) {  return [...new Set(arr)];}
五、最有趣的去重:filter + indexOf去重

利用filter + indeOf其实也可以实现去重,而且是一行代码搞定,非常有趣。

核心的思路就是,在用filter遍历时,用indexOf去查找当前遍历项的索引,如果查找到的索引与此时filter遍历的index一样,那么说明在当前项之前是没出现与当前项相同的重复项的,所以该项需要保留,反之则需要剔除,而正好filter自带过滤功能,返回true保留该项,返回false剔除该项。

function unique(arr) {  return arr.filter((item, index) => arr.indexOf(item) === index);}const arr = [1,1,2,2,3,4,4,5,5,6];console.log(unique(arr));// 输出:[ 1, 2, 3, 4, 5, 6 ]
六、最骚的去重:JSON.stringify + Set + JSON.parse

想不到吧,利用JSON.stringify + Set + JSON.parse居然也可以实现去重,这应该是最骚的去重方式了,而且天然就支持对存放数组的对象进行去重。

去重的核心思路如下:

利用JSON.stringify可以将数组每一项存储的对象或者普通值都进行序列化,将原数组转为一个字符串数组;

利用Set对字符串数组进行去重;

利用JSON.parse对数组每一项进行反序列化。

这里注意,如果两个对象存的key和value都相同,则他们序列化的结果也相同,所以说这种方式是天然支持对对象进行去重的。

function unique(arr) {    return [...new Set(arr.map(t => JSON.stringify(t)))].map(s => JSON.parse(s));}const arr = [1,1,2,2,3,4,4,5,5,6];const arr1 = [{ a:1 }, { a:1 }, { a:2 }, { a:2 }, { a:3 }, { a:3 }, { a:4 }, { a:4 }, { a:5 }, { a:5 } ]console.log(unique(arr));console.log(unique(arr1));/** * 打印结果: * [ 1, 2, 3, 4, 5, 6 ] * [ { a: 1 }, { a: 2 }, { a: 3 }, { a: 4 }, { a: 5 } ] */

以上介绍了6种JavaScript去重方式,我平常最常用的就是第四种,也就是展开运算符 + Set去重,不知道大家在平时实际开发中更喜欢哪一种去重方式呢,或者还有没有其它的更好去重方式呢?

标签: #js数组能存多少数据啊