前端学习:教程&开发模块化/规范化/工程化/优化&工具/调试&值得关注的博客/Git&面试-前端资源汇总

欢迎提issues斧正:数组去重

JavaScript-数组去重由慢到快由繁到简演化

indexOf去重

Array.prototype.unique1 = function() {var arr = [];for (var i = 0; i < this.length; i++) {var item = this[i];if (arr.indexOf(item) === -1) {arr.push(item);}}return arr;
}
[1,2,3,'4',3,4,3,1,'34',2].unique1(); //[1, 2, 3, "4", 4, "34"]//filter+indexOf写法,箭头函数为ES6新写法。
Array.prototype.unique1 = function() {return this.filter((item, index, arr) => arr.indexOf(item) === index);
}

indexOf的思想就是遍历一个数组的字符,判断这个字符在另一个数组存不存在,不存在就把这个字符也弄一个到结果数组里去。在 IE6-8 下,数组的 indexOf 方法还不存在(虽然这已经算有点古老的话题了O(∩_∩)O~),但是,程序员就要写一个indexOf方法:

var indexOf = [].indexOf ? function(arr, item) {return arr.indexOf(item);
} :
function indexOf(arr, item) {for (var i = 0; i < arr.length; i++) {if (arr[i] === item) {return i;}}return -1;
}Array.prototype.unique2 = function() {var arr = [];for (var i = 0; i < this.length; i++) {var item = this[i];if (arr.indexOf(item) === -1) {arr.push(item);}}return arr;
}
[1,2,3,'4',3,4,3,1,'34',2].unique2(); //[1, 2, 3, "4", 4, "34"]

indexOf还可以以这样的去重思路:判断当前字符在数组中出现的位置是不是第一次出现的位置,如果是就把字符放到结果数组中。在去重过程中,原数组都是不变的。

Array.prototype.unique3 = function(){var arr = [this[0]]; for(var i = 1; i < this.length; i++){if (this.indexOf(this[i]) == i){arr.push(this[i]);} }return arr;
}
[1,2,3,'4',3,4,3,1,'34',2].unique3(); //[1, 2, 3, "4", 4, "34"]

hash去重

以上indexOf正确性没问题,但性能上,两重循环会降低性能。那我们就用hash。

Array.prototype.unique4 = function() {var arr = [];var hash = {};for (var i = 0; i < this.length; i++) {var item = this[i];var key = typeof(item) + itemif (hash[key] !== 1) {arr.push(item);hash[key] = 1;}} return arr;
}
[1,2,3,'4',3,4,3,1,'34',2].unique4(); //[1, 2, 3, "4", 4, "34"]

hash去重的核心是构建了一个 hash 对象来替代 indexOf。以空间换时间。注意在 JavaScript 里,对象的键值只能是字符串,因此需要var key = typeof(item) + item 来区分数值 1 和字符串 '1' 等情况。(当然,ES6提供了Map数据结构。它类似于对象,也是键值对的集合,但是“键”的范围不限于字符串,各种类型的值(包括对象)都可以当作键。也就是说,Object结构提供了“字符串—值”的对应,Map结构提供了“值—值”的对应,是一种更完善的Hash结构现。)

那如果你想要'4' 和 4 被认为是相同的话(其他方法同理)

Array.prototype.unique5 = function(){var arr=[];var hash={};for(var i=0,len=this.length;i<len;i++){if(!hash[this[i]]){ arr.push(this[i]);hash[this[i]]=true;}}return arr;
}
[1,2,3,'4',3,4,3,1,'34',2].unique5(); //[1, 2, 3, "4", "34"]

排序后去重

Array.prototype.unique6 = function(){this.sort();var arr = [this[0]];for(var i = 1; i < this.length; i++){if( this[i] !== arr[arr.length-1]){arr.push(this[i]);}}return arr;
}
[1,2,3,'4',3,4,3,1,'34',2].unique6(); //[1, 2, 3, "34", "4", 4]

先把数组排序,然后比较相邻的两个值,排序的时候用的JS原生的sort方法,所以非常快。而这个方法的缺陷只有一点,比较字符时按照字符编码的顺序进行排序。所以会看到10排在2前面这种情况。不过在去重中不影响。不过,解决sort的这个问题,是sort方法接受一个参数,这个参数是一个方法:

function compare(value1,value2) {if (value1 < value2) {return -1;} else if (value1 > value2) {return 1;} else {return 0;}
}
[1,2,5,2,10,3,20].sort(compare);  //[1, 2, 2, 3, 5, 10, 20]

Set去重

ES6提供了新的数据结构Set。它类似于数组,但是成员的值都是唯一的,没有重复的值。现在浏览器正在全面支持,服务端的node也已经支持。

Array.prototype.unique7 = function(){return [...new Set(this)];
}
[1,2,3,'4',3,4,3,1,'34',2].unique7(); //[1, 2, 3, "4", 4, "34"]

方法库

推荐一个方法库Underscore.js,在node或浏览器js中都很受欢迎。

const _ = require('underscore');
_.uniq([1, 2, 1, 3, 1, 4]);  //[1, 2, 3, 4]

测试时间

以上方法均可以用一个简单的方法去测试一下所耗费的时间,然后对各个方法做比较择优:

console.time("test");
[1,2,3,'4',3,4,3,1,'34',2].unique7();
console.timeEnd("test");
==> VM314:3 test: 0.378ms

让数据变得大一点,就随机创建100万个数:

var arr = [];
var num = 0;
for(var i = 0; i < 1000000; i++){num = Math.floor(Math.random()*100);arr.push(num);
}
console.time("test");
arr.unique7();
console.timeEnd("test");
==> VM325:3 test: 108025.815ms (比较数目越多,差距越大,更好选择)

我们平时使用数组去重的地方,视业务不同,需求量不一样。但使用的方法则可以视业务场景而选择一个正确的合适的方法来写代码。更重要的是我们的代码要写来让别人看得懂...写晦涩难懂的代码切不做注释只是装得一手好逼。。。

温故js系列(7)-数组去重由慢到快由繁到简相关推荐

  1. 温故js系列(11)-BOM

    前端学习:教程&开发模块化/规范化/工程化/优化&工具/调试&值得关注的博客/Git&面试-前端资源汇总 欢迎提issues斧正:BOM JavaScript-BOM ...

  2. 去重查询_《前端算法系列》数组去重

    虽然算法在前端开发中很少会得以使用,但是了解常用的算法,熟悉各种算法的性能和优劣,将会让你在前端的道路上走的更远. 前言 文中所有代码位于位于此代码仓库中,大家可以下载代码进行学习.推敲和改进.另,如 ...

  3. js 简单的数组去重13种方法

    js 数组去重 十几种数组去重的方法,有的去重构思可以大致相同. 上篇的几种数组去重 https://blog.csdn.net/weixin_47988564/article/details/106 ...

  4. es6 数组去重_《前端算法系列》数组去重

    虽然算法在前端开发中很少会得以使用,但是了解常用的算法,熟悉各种算法的性能和优劣,将会让你在前端的道路上走的更远. 前言 文中所有代码位于位于此代码仓库中,大家可以下载代码进行学习.推敲和改进.另,如 ...

  5. js最简单数组去重_js数组去重最常用的5种方法

    今天来聊一聊js数组去重的一些方法,包括一些网上看到的和自己总结的,总共5种方法,希望对大家有帮助. 第一种:遍历数组法 这种方法最简单最直观,也最容易理解,代码如下: 这种方法很好理解,利用了ind ...

  6. JS篇之数组去重、排序、求最值的几种方法封装

    一.数组去重(仅列五种常见) 1. es6的 set function unique(arr){return [...new Set(arr)] } 2. 数组的indexOf function un ...

  7. js、ES6数组去重

    数组去重的方式 提示:reduce,Array.from(new Set(array)),扩展运算符 -,map方法等 reduce去重 // 数组对象去重 let obj = {}; let arr ...

  8. 每日题(js):数组去重,数组排序

    题目:js 数组的去重,排序,各有几种方式,分别是怎么实现的? 去重: // 去重 filter/indexOf let arr = [1, 1, 0, 0, 8, 8, 6, 6, 5, 5] le ...

  9. 温故js系列(4)-运算符详解

    前端学习:教程&开发模块化/规范化/工程化/优化&工具/调试&值得关注的博客/Git&面试-前端资源汇总 欢迎提issues斧正:运算符 JavaScript-运算符 ...

最新文章

  1. JZOJ #4722 跳楼机 (最短路模型的完美转化)
  2. 智慧农业物联网云平台方案
  3. 开课啦! dubbo-go 微服务升级实战
  4. 判断使用设备是PC还是phone
  5. python 温度转换程序_Python程序将米转换为码
  6. java桥_JAVA 桥模式
  7. 洛谷 P4660 bzoj 1168 [ Baltic OI 2008 ] 手套 —— 分析+单调栈
  8. 低脂肪肉能帮你减肥吗?
  9. php分析图片中水印的位置,图像处理技术之图片添加水印
  10. php 获取用户名,php 获取Feedburner的用户名示例
  11. 【PAT】2020年秋季考试划水准备贴
  12. python英文文本词性分析
  13. MySQL从入门到精通之sql语言---(9月2日更新)
  14. 从借呗借了100元,看背后的系统架构
  15. R语言威尔科克森符号秩和(Wilcoxon Signed Rank statistic)分布
  16. 从数字城市迈向智能城市
  17. 决策树应用实例①——泰坦尼克号分类
  18. 【JavaScript】JS 的执行
  19. IntelliJ IDEA设置编码格式
  20. 一位微信小程序萌新的学渣笔记(三)基础语法之常见组件

热门文章

  1. 不安装oracle使用exp命令
  2. mysql 过滤单引号_python实现mysql的单引号字符串过滤方法
  3. 拆除指令怎么设置_siri叫你起床啦~早安等快捷指令你会用了吗i
  4. Cannot forward after response has been committed问题解决及分析
  5. 基于《悉尼协议》框架下Java课程案例教学研究
  6. Java案例:几种方式拷贝文件的耗时比较
  7. 在Eclipse里搭建Scala开发环境
  8. 查询:使用多表连接查询数据
  9. 2017.4.16 级数求和 思考记录
  10. SSM框架入门学习记录