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

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

ruisui883个月前 (02-03)技术分析14

在平时的前端开发中,常常会遇到数组去重的场景,比如表单校验重复的数据,有重复数据不让提交,或者对后端返回的数据进行去重展示等,同时它也是一道经典的前端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 数组去重)” 的相关文章

vue3父子组件传对象,子组件访问修改父组件对象中的属性值

在Vue 3中,父子组件之间的数据传输通常通过props和emit进行。父组件可以通过props向下传递数据给子组件,子组件则可以通过emit向上通知父组件更新数据。如果需要在子组件中修改父组件对象中的属性值,可以使用一个名为ref的Vue 3新特性。以下是一个示例,演示了如何在Vue 3中实现父子...

Gitlab+Jenkins通过钩子实现自动部署web项目,图文详细教程

扩展参考:Jenkins+Gitlab通过脚本自动部署回滚web项目至集群 一:基础环境介绍及准备1):Gitlab服务器:ubuntu 192.168.152.131 ---参考搭建:Linux安装gitlab,docker安装gitlab教程2):Jenkins服务器:ubunu 192.168...

摄影后期必看 | PS插件camera raw 16.4教程 | 范围蒙版

范围蒙版Camera Raw 【蒙版】模块中提供了三个范围蒙版工具,可以通过特定的范围来创建蒙版。此次新增的【范围蒙版】大大加强了acr插件对局部调整的能力。点击下拉小箭头可以看到【颜色范围】,可用于快速选择想要编辑的颜色。快捷键:Shift + C【明亮度范围】,可用于快速选择想要调整的明亮度。快...

Vue Router 4 路由操作 - 路由导航

路由导航分为 声明式导航 和 编程式导航。通过 <router-link to="..."> 标签跳转的方式为声明式导航。通过 路由实例对象(router.push(...))跳转的为编程式导航。导航到不同的位置想要导航到不同的URL,使用 router.push 方法。...

html5迁移到微信小程序的 方法 亲测可用

切图网习惯于在做小程序之前先做成html5+vuejs的形式,因为html5切图是我们比较熟悉的方式,而且有专业的工具 以及浏览器调试也会更加的方便 灵活,效率高,而且html5的方式可以方便预览看效果,方便调整,当html5页面做好确认没问题以后 再转成小程序或者官方出品wepy的方式,这个时候就...

vue父组件修改子组件的值(通过调用子组件的方法)

props只支持第一次加载这个组件的时候获取父组件的值,后续修改父组件的值得时候子组件并不会动态的更改。然而我们想要通过父组件修改子组件的值要怎么做呢?可以通过ref的方式调用子组件的方法改变子组件的值。子组件<template><div><span>{{data...