当前位置:首页 > 技术分析 > 正文内容

JavaScript数组去重花样大赏(js 数组去重)

ruisui885个月前 (02-03)技术分析22

在平时的前端开发中,常常会遇到数组去重的场景,比如表单校验重复的数据,有重复数据不让提交,或者对后端返回的数据进行去重展示等,同时它也是一道经典的前端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^2)降为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去重,不知道大家在平时实际开发中更喜欢哪一种去重方式呢,或者还有没有其它的更好去重方式呢?

扫描二维码推送至手机访问。

版权声明:本文由ruisui88发布,如需转载请注明出处。

本文链接:http://www.ruisui88.com/post/687.html

分享给朋友:

“JavaScript数组去重花样大赏(js 数组去重)” 的相关文章

Linux发行版Debian推出12.2及11.8版本,修复多个安全问题

IT之家 10 月 9 日消息,Debian 是最古老的 GNU / Linux 发行版之一,也是许多其他基于 Linux 的操作系统的基础,包括 Ubuntu、Kali、MX 和树莓派 OS 等,近日 Debian 推出了 12.2 和 11.8 版本,主要修复了多个安全问题。▲ 图源 Debia...

手把手教你Vue之父子组件间通信实践讲解【props、$ref 、$emit】

组件是 vue.js 最强大的功能之一,而组件实例的作用域是相互独立的,这就意味着不同组件之间的数据无法相互引用。那么组件间如何通信,也就成为了vue中重点知识了。这篇文章将会通过props、$ref和 $emit 这几个知识点,来讲解如何实现父子组件间通信。转载链接:https://www.jia...

「Git迁移」三行命令迁移Git包含提交历史,分支,tag标签等信息

问题描述:公司需要将一个git远程服务器的全部已有项目迁移到一台新服务器的Gitlab中,其中需要包含全部的提交纪录,已有的全部分支与全部打tag标签,目前此工作已全部迁移完毕,特此记录一下操作步骤环境描述:1. 要迁移的远程Git:Gitblit2. 迁移目的Git:Gitlab3. 暂存代码的P...

虚幻引擎5.5发布

IT之家 11 月 13 日消息,虚幻引擎 5.5 现已发布。据介绍,新版本虚幻引擎在动画创作、虚拟制作和移动游戏开发方面取得进步;渲染、摄像机内视觉特效和开发人员迭代等领域的部分功能已可用于生产。IT之家整理部分功能亮点如下:动画Sequencer增强虚幻引擎的非线性动画编辑器 Sequencer...

雅马哈TMAX 560 TECH MAX 外媒深度测评

应雅马哈(Yamaha)的邀请,在葡萄牙埃斯托里尔对全新的Yamaha TMAX 560 Tech Max踏板车进行了测试,在这里TMAX 560 Tech Max售价为11649英镑。雅马哈TMAX长期以来一直站在踏板车的顶端,就声誉和知名度而言,它是当之无愧的大踏板界NO.1。2020 TMAX...

三、Uni-app + vue3 页面如何跳转及传参?

Vue 项目往往需要使用 vue-router 插件,刚开始入门 Uni-app + Vue3 项目的同学,会不会想着路由使用 vue-router V4 版本不就可以了吗?不怕大家笑话,我就是这样想的,毕竟我是第一次使用 Uni-app ,由于孕期记性贼差,所以我决定写成笔记,加深记忆。uni-a...