JS数组去重的9种方法(包括去重NaN和复杂数组类型)
其实网上已经有很多js数组的去重方法,但是我看了很多篇并自己通过代码验证,发现都有一些缺陷,于是在研究多篇代码之后,自己总结了9种方法,如果有哪里不对请及时纠正我哈~
转载请表明出处
测试代码
let arr1 = [3, 1, [1], 1, [1], true, true, {}, '1', NaN, undefined, NaN, undefined, {}, null, null];
let arr2 = [];
for (let i = 0; i < 100000; i++) {
arr2.push(0 + Math.floor((100000 - 0 + 1) * Math.random()));
}
// 封装在Array的原型对象会更好,this就是指向调用该方法的数组
Array.prototype.unique = function () {
//...
}
console.log(arr1.unique()); // 测试去重效果
console.time('test');
console.log(arr2.unique()); // 测试去重时间
console.timeEnd('test');
备注:
arr1- 包含了两个相同的数组[1]、并且数组[1]和1的顺序打乱(这是为了突出sort的弊端)
- 有两个NaN、两个undefined、两个null等等,把平时会用到的数据类型都写上来了
- 有两个相同的空对象{}
arr2- 放入10万个随机纯数字,大的数据量才能看出时间的差别
- 对于纯number的数组,以下所有方法(包括网上的各种方法)都是可行的
- 正确去重的
arr1:[3, 1, [1], true, {}, '1', NaN, undefined, null],length = 9。
一、不能正确去重NaN和Object的方法
1. 两种for循环 + splice(耗时最长)
Array.prototype.unique = function () {
for (let i = 0; i < this.length; i++) {
for (let j = i + 1; j < this.length; j++) {
if (this[i] === this[j]) {
this.splice(j, 1);
j--;
}
}
}
return this;
}
- test: 21208.31396484375 ms(花了21s....)
不能去重NaN和复杂数组类型(比如 Object 和 Array )
2. forEach + indexOf
Array.prototype.unique = function () {
let newArr = [];
this.forEach((item) => {
if (newArr.indexOf(item) === -1) {
newArr.push(item);
}
})
return newArr;
}
结果同上,建议大家自己运行一下(就是懒得截图
- test: 4104.52294921875 ms
不能去除重复的NaN和复杂数据类型
- 原因:indexOf 认为 NaN 不等于 NaN
3. filter + indexOf
Array.prototype.unique = function () {
return this.filter((item, index) => {
// 利用indexOf检测元素在数组中第一次出现的位置是否和元素现在的位置相等,如果不等则说明该元素是重复元素
return this.indexOf(item) === index;
})
}
- test: 5682.358154296875 ms
不能去掉重复的复杂数据类型,同时还会去掉所有的NaN
- 原因:indexOf 认为 NaN 不等于 NaN ,所以也就不认为他们是重复元素。
4. for + sort(sort有问题)
Array.prototype.unique = function () {
let newArr = [];
this.sort();
for (let i = 0; i < this.length; i++) {
if (this[i] !== this[i + 1]) {
newArr.push(this[i]);
}
}
return newArr;
}
- test: 61.96484375 ms
带 sort 方法的只对纯number或者纯string类型有效,它无法区分1和'1',因为它是在将元素转换为字符串,然后比较它们的UTF-16代码单元值序列时构建的。
这时候如果有一段这样的排序[1, '1', 1],再用前后比较的方法去重就会出现问题
在这里他还会把undefined也全都去掉,原因是
sort()方法排序后,undefined排在最后,而最后一个undefined要和this[length]进行比较,而这个值并不存在,而也是undefined,就会认为他们是同一个值。
同样它不能去重NaN和复杂数据类型,但是耗时是在方法里面是最少之一了(都之一了用别的不香吗)
5. sort + reduce(sort有问题)
Array.prototype.unique = function () {
return this.sort().reduce((init, cur) => {
if (init.length === 0 || init[init.length - 1] !== cur) {
init.push(cur);
}
return init;
}, []);
}
- test: 66.679931640625 ms
同样他也是用sort排序再前后比较,也是傻逼玩意
比上面那个方法好一点点,还能正确去重undefined(我觉得前后比较这种方法着实不靠谱)
同样也不能去重NaN和复杂数组类型,耗时也是在方法里面是最少之一。
二、能正确去重NaN,不能去重复杂数据类型
1. forEach + includes
Array.prototype.unique = function () {
let newArr = [];
this.forEach((item) => {
if (!newArr.includes(item)) {
newArr.push(item);
}
})
return newArr;
}
- 4181.393798828125 ms
可以去掉重复的NaN,但是不能去掉重复的复杂数据类型
- includes 认为 NaN === NaN 为 true
2. forEach + map
Array.prototype.unique = function () {
let map = new Map();
let newArr = new Array();
this.forEach((item) => {
if (!map.has(item)) {
map.set(item, 1);
newArr.push(item);
}
});
return newArr;
}
同上,不截图了,自行运行一哈~
- test: 27.030029296875 ms
可以去NaN,不能去重复杂数组类型,运行速度快,耗时最少的方法之一
3. Set
Array.prototype.unique = function () {
return [...new Set(this)];
}
同上,不截图了
- test: 31.197021484375 ms
可以去掉重复的NaN,但是不能去掉重复的复杂数据类型,运行速度快,耗时最少的方法之一,代码最短!
三、可以去掉NaN和复杂数据类型的!
1. filter + hasOwnProperty + JSON.stringify
Array.prototype.unique = function () {
let obj = {};
return this.filter(function (item, index, arr) {
return obj.hasOwnProperty(typeof item + JSON.stringify(item)) ? false : (obj[typeof item + JSON.stringify(item)] = true);
});
}
- test: 126.3359375 ms
可以去掉重复的NaN和重复的复杂数据类型
在object中,key如果是number类型,它会自动转换成string类型,所以{1:1}和{"1":1}是相等的,这不是这个方法的缺陷,这是Oject的缺陷
而{1: 1}和{1: "1"}这种value值不同的可以正确区分开。
- 该方法的核心:以
typeof item元素类型+item的字符串作为key - 有些文章写的方法是直接使用item,让obj自行隐式转换成字符串
- 考虑到obj的字符串都为'[object Object]',这里使用JSON.stringify(item),就可以保存不同的obj字符串
后记
其实上面的方法不外乎三种:
- 利用for、forEach、filter遍历,再利用indexOf、includes等方法判断是否重复;
- 利用Set数据结构的特性;
- 利用obj或者map的key不能重复的特性。
实际开发应该更多的是纯数字的去重(吧?),而且也用不上这么多方法。能记住两三种根据实际情况的需求来选择就行了(吧?)
但作为一个正在找工作的应届生,不得不总结多几种方法,这样面试官问起来的时候才不至于口哑无言嘿嘿嘿
(完)
JS数组去重的9种方法(包括去重NaN和复杂数组类型)的更多相关文章
- js中数组去重的几种方法
js中数组去重的几种方法 1.遍历数组,一一比较,比较到相同的就删除后面的 function unique(arr){ ...
- JS实现数组去重的6种方法总结
方法一: 双层循环,外层循环元素,内层循环时比较值,如果有相同的值则跳过,不相同则push进数组. Array.prototype.distinct = function(){ var arr = t ...
- JavaScript中数组去重的几种方法
JavaScript中数组去重的几种方法 正常情况下,数据去重的工作一般都是由后端同事来完成的,但是前端也要掌握好处理数据的能力,万一去重的工作交给我们大前端处理,我们也不能怂呀.现在我总结了一些去重 ...
- PHP中数组合并的两种方法及区别介绍
PHP数组合并两种方法及区别 如果是关联数组,如下: 复制代码代码如下: $a = array( 'where' => 'uid=1', 'order' => 'uid', ); $b = ...
- ajax请求参数中含有特殊字符"#"的问题 (另附上js编码解码的几种方法)
使用ajax向后台提交的时候 由于参数中含有# 默认会被截断 只保留#之前的字符 json格式的字符串则不会被请求到后台的action 可以使用encodeURIComponent在前台进行编码, ...
- List集合对象去重及按属性去重的8种方法-java基础总结系列第六篇
最近在写一些关于java基础的文章,但是我又不想按照教科书的方式去写知识点的文章,因为意义不大.基础知识太多了,如何将这些知识归纳总结,总结出优缺点或者是使用场景才是对知识的升华.所以我更想把java ...
- js数组去重的4种方法
js数组去重,老生长谈,今天对其进行一番归纳,总结出来4种方法 贴入代码前 ,先对浏览器Array对象进行支持indexOf和forEach的polyfill Array.prototype.inde ...
- js数组去重的几种方法
1.遍历数组法 最简单的去重方法, 实现思路:新建一新数组,遍历传入数组,值不在新数组就加入该新数组中:注意点:判断值是否在数组的方法“indexOf”是ECMAScript5 方法,IE8以下不支持 ...
- Js 数组去重的几种方法总结
去重是开发中经常会碰到的一一个热点问题,不过目前项目中碰到的情况都是后台接口使用SQL去重,简单高效,基本不会让前端处理去重.那么前端处理去重会出现什么情况呢?假如每页显示10条不同的数 ...
随机推荐
- 如何检测ASP中的浏览器。NET与浏览器文件
介绍 ASP.NET是一个用于使用Web表单.MVC.Web API和SignalR(这是官方定义)构建Web应用程序的高生产力框架.它是在.net框架上开发RESTful应用程序或使用HTML.CS ...
- 用Python爬取B站、腾讯视频、爱奇艺和芒果TV视频弹幕!
众所周知,弹幕,即在网络上观看视频时弹出的评论性字幕.不知道大家看视频的时候会不会点开弹幕,于我而言,弹幕是视频内容的良好补充,是一个组织良好的评论序列.通过分析弹幕,我们可以快速洞察广大观众对于视频 ...
- JavaCV FFmpeg H264编码
上次成功通过FFmpeg采集摄像头的YUV数据,这次针对上一次的程序进行了改造,使用H264编码采集后的数据. (传送门) JavaCV FFmpeg采集摄像头YUV数据 采集摄像头数据是一个解码过程 ...
- IDEA推送docker镜像到私服/利用dockerfile-maven-plugin插件在springboot中上传镜像到远程的docker服务器、远程仓库
利用dockerfile-maven-plugin插件在springboot中上传镜像到远程仓库 这篇文章讲解在开发工具中把打包好的jar编译成docker镜像,上传到远程的docker服务 ...
- 微服务通信之feign集成负载均衡
前言 书接上文,feign接口是如何注册到容器想必已然清楚,现在我们着重关心一个问题,feign调用服务的时候是如何抉择的?上一篇主要是从读源码的角度入手,后续将会逐步从软件构架方面进行剖析. 一.R ...
- PHP的学习(提前学习了,业余爱好) (一)
一个函数一个函数地堆 strstr()函数 在本地测试的时候,代码与显示如下 1.代码: <?php echo strstr("I love Shanghai!123",&q ...
- 警惕char类型直接相加
今天在写某个程序需要对两个数字字符串进行相加操作,比如字符串1:12345,字符串2:23456.需要1和2相加.2和3相加.就是两个字符相同位置的数进行相加. 这个一看很好完成,写一个for,然后取 ...
- SessionStorage、LocalStorage详解
转载请注明出处:葡萄城官网,葡萄城为开发者提供专业的开发工具.解决方案和服务,赋能开发者. 原文出处:https://blog.bitsrc.io/sessionstorage-and-localst ...
- 手把手教你用 Spring Boot搭建一个在线文件预览系统!支持ppt、doc等多种类型文件预览
昨晚搭建环境都花了好一会时间,主要在浪费在了安装 openoffice 这个依赖环境上(Mac 需要手动安装). 然后,又一步一步功能演示,记录,调试项目,并且简单研究了一下核心代码之后才把这篇文章写 ...
- c语言版去除源代码注释
去除代码中注释需要注意下面几点 首先注释有"/*"开始到"*/"结束的多行或单行注释 其次还有"//"这种单行注释 另外还需要注意双引号和单 ...