【JavaScript】数组去重

话说面试常会碰到面试官会问JavaScript实现数组去重的问题,最近刚好在学习有关于JavaScript数组相关的知识,趁此机会整理了一些有关于JavaScript数组去重的方法。

下面这些数组去重的方法是自己收集和整理的,如有不对希望指正文中不对之处。

双重循环去重


这个方法使用了两个for循环做遍历。整个思路是:

  • 构建一个空数组用来存放去重后的数组
  • 外面的for循环对原数组做遍历,每次从数组中取出一个元素与结果数组做对比
  • 如果原数组取出的元素与结果数组元素相同,则跳出循环;反之则将其存放到结果数组中

代码如下:

Array.prototype.unique1 = function () { 
 // 构建一个新数组,存放结果 var newArray = [this[0]]; 
 // for循环,每次从原数组中取出一个元素 
 // 用取出的元素循环与结果数组对比 
 for (var i = 1; i < this.length; i++) { 
  var repeat = false; 
  for (var j=0; j < newArray.length; j++) { 
  // 原数组取出的元素与结果数组元素相同 
   if(this[i] == newArray[j]) { 
    repeat = true; break; 
   } 
  } 
  if(!repeat) { 
  // 如果结果数组中没有该元素,则存放到结果数组中     
   newArray.push(this[i]);
  } 
 } 
 return newArray;
}

假设我们有一个这样的数组:

var arr = [1,2,3,4,'a','b',1,3,4,56,32,34,2,'b','c',5,'1','2'];
arr.unique1(); // [1, 2, 3, 4, "a", "b", 56, 32, 34, "c", 5]

据说这种方法比较耗时,费性能。简单做个测试(测试方法写得比较拙逼):

function test () { 
 var arr = []; 
 for (var i = 0; i < 1000000; i++) {   
  arr.push(Math.round(Math.random(i) * 10000)); 
 } 
 doTest(arr, 1);
}
function doTest(arr, n) { 
 var tStart = (new Date()).getTime(); 
 var re = arr.unique1(); 
 var tEnd = (new Date()).getTime(); 
 console.log('双重循环去重方法使用时间是:' + (tEnd - tStart) + 'ms'); 
 return re;
}
test();

在Chrome控制器运行上面的代码,测试双重循环去重所费时间:11031ms。

上面的方法可以使用forEach()方法和indexOf()方法模拟实现:

function unique1() { 
 var newArray = []; 
 this.forEach(function (index) { 
  if (newArray.indexOf(index) == -1) { 
   newArray.push(index); 
  } 
 }); 
 return newArray;
}

通过unique1.apply(arr)或unique1.call(arr)调用。不过这种方法效率要快得多,同样的上面测试代码,所费时间5423ms,几乎快了一半。

排序遍历去重


先使用sort()方法对原数组做一个排序,排完序之后对数组做遍历,并且检查数组中的第i个元素与结果数组中最后一个元素是否相同。如果不同,则将元素放到结果数组中。

Array.prototype.unique2 = function () { 
 // 原数组先排序 
 this.sort(); 
 // 构建一个新数组存放结果 
 var newArray = []; 
 for (var i = 1; i < this.length; i++) { 
  // 检查原数中的第i个元素与结果中的最后一个元素是否相同 
  // 因为排序了,所以重复元素会在相邻位置 
  if(this[i] !== newArray[newArray.length - 1]) { 
   // 如果不同,将元素放到结果数组中 
   newArray.push(this[i]); 
  } 
 } 
 return newArray;
}

例如:

var arr = [1,2,3,4,'a','b',1,3,4,56,32,34,2,'b','c',5,'1','2'];
arr.unique2(); // ["1", 1, 2, "2", 3, 32, 34, 4, 5, 56, "a", "b", "c"]

这种方法有两个特色:
去重后的数组会做排序,主要是因为原数在去重前做了排序去重后的数组,与数字相同的数字字符无法区分,比如'1'和1

使用同样的方法,测试所费时间:1232ms。

对象键值对法


这种去重方法实现思路是:
创建一个JavaScript对象以及新数组使用for循环遍历原数组,每次取出一个元素与JavaScript对象的键做对比
如果不包含,将存入对象的元素的值推入到结果数组中,并且将存入object
对象中该属性名的值设置为1

代码如下:

Array.prototype.unique3 = function () { 
 // 构建一个新数组存放结果 
 var newArray = []; 
 // 创建一个空对象 
 var object = {}; 
 // for循环时,每次取出一个元素与对象进行对比 
 // 如果这个元素不重复,则将它存放到结果数中 
 // 同时把这个元素的内容作为对象的一个属性,并赋值为1, 
 // 存入到第2步建立的对象中 
 for (var i = 0; i < this.length; i++){ 
  // 检测在object对象中是否包含遍历到的元素的值 
  if(!object[typeof(this[i]) + this[i]]) { 
  // 如果不包含,将存入对象的元素的值推入到结果数组中
   newArray.push(this[i]); 
   // 如果不包含,存入object对象中该属性名的值设置为1 
   object[typeof(this[i]) + this[i]] = 1; 
  } 
 } 
 return newArray;
}

运行前面的示例:

var arr = [1,2,3,4,'a','b',1,3,4,56,32,34,2,'b','c',5,'1','2'];
arr.unique3(); // [1, 2, 3, 4, "a", "b", 56, 32, 34, "c", 5, "1", "2"]

同样的,不同的键可能会被误认为一样;例如: a[1]、a["1"]。这种方法所费时间:621ms。 这种方法所费时间是最短,但就是占用内存大一些。

除了上面几种方法,还有其他几种方法如下:

// 方法四
Array.prototype.unique4 = function () { 
 // 构建一个新数组存放结果 
 var newArray = []; 
 // 遍历整个数组 
 for (var i = 0; i < this.length; i++) { 
  // 遍历是否有重复的值 
  for (j = i + 1; j < this.length; j++) { 
   // 如果有相同元素,自增i变量,跳出i的循环 
   if(this[i] === this[j]) { 
    j = ++i; 
   }  
  } 
  // 如果没有相同元素,将元素推入到结果数组中  
  newArray.push(this[i]); 
 } 
 return newArray;
}

Chrome测试结果

var arr = [1,2,3,4,'a','b',1,3,4,56,32,34,2,'b','c',5,'1','2'];
arr.unique4(); // ["a", 1, 3, 4, 56, 32, 34, 2, "b", "c", 5, "1", "2"]

同样的,1和'1'无法区分。

// 方法五
Array.prototype.unique5 = function () { 
 // 构建一个新数组存放结果 
 var newArray = []; 
 // 遍历整个数组 
 for (var i = 0; i < this.length; i++) { 
  // 如果当前数组的第i值保存到临时数组,那么跳过 
  var index = this[i]; 
  // 如果数组项不在结果数组中,将这个值推入结果数组中 
  if (newArray.indexOf(index) === -1) { 
   newArray.push(index); 
  } 
 } 
 return newArray;
}

Chrome测试结果:

var arr = [1,2,3,4,'a','b',1,3,4,56,32,34,2,'b','c',5,'1','2'];
arr.unique6(); // [1, 2, 3, 4, "a", "b", 56, 32, 34, "c", 5, "1", "2"]

同样的,类似于1和'1'无法区分。所费时间:14361ms。

// 方法六
Array.prototype.unique6 = function () { 
 return this.reduce(function (newArray, index) {   
  if(newArray.indexOf(index) < 0) { 
   newArray.push(index); 
  } 
  return newArray; 
 },[]);
}

测试结果如下:

var arr = [1,2,3,4,'a','b',1,3,4,56,32,34,2,'b','c',5,'1','2']; 
arr.unique6(); // [1, 2, 3, 4, "a", "b", 56, 32, 34, "c", 5, "1", "2"]

所费时间:16490ms。

// 方法七
Array.prototype.unique7 = function(){ 
 var newArray; 
 newArray = this.filter(function (ele,i,arr) { 
  return arr.indexOf(ele) === i; }); 
  return newArray;
}

测试结果:

var arr = [1,2,3,4,'a','b',1,3,4,56,32,34,2,'b','c',5,'1','2']; 
arr.unique6(); // [1, 2, 3, 4, "a", "b", 56, 32, 34, "c", 5, "1", "2"]

所费时间:13201ms。
方法虽然很多种,但相比下来,下面这种方法是较为优秀的方案:

//方法三
Array.prototype.unique3 = function () { 
 // 构建一个新数组存放结果 
 var newArray = []; 
 // 创建一个空对象 
 var object = {}; 
 // for循环时,每次取出一个元素与对象进行对比 
 // 如果这个元素不重复,则将它存放到结果数中 
 // 同时把这个元素的内容作为对象的一个属性,并赋值为1, 
 // 存入到第2步建立的对象中 
 for (var i = 0; i < this.length; i++){ 
  // 检测在object对象中是否包含遍历到的元素的值 
  if(!object[typeof(this[i]) + this[i]]) { 
  // 如果不包含,将存入对象的元素的值推入到结果数组中
   newArray.push(this[i]); 
   // 如果不包含,存入object对象中该属性名的值设置为1 
   object[typeof(this[i]) + this[i]] = 1; 
  } 
 } 
 return newArray;
}

但在ES6去重还有更简单,更优化的方案,比如:

// ES6
function unique (arr) { 
 const seen = new Map() 
 return arr.filter((a) => !seen.has(a) && seen.set(a, 1))
}
// or
function unique (arr) { 
 return Array.from(new Set(arr))
}

转载自:http://www.w3cplus.com/javascript/remove-duplicates-from-javascript-array.html

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,847评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,208评论 1 292
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,587评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,942评论 0 205
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,332评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,587评论 1 218
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,853评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,568评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,273评论 1 242
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,542评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,033评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,373评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,031评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,073评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,830评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,628评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,537评论 2 269

推荐阅读更多精彩内容

  • Why underscore (觉得这部分眼熟的可以直接跳到下一段了...) 最近开始看 underscore.j...
    韩子迟阅读 1,363评论 0 57
  • 偶然和同事谈到面试的Javascript问题,其中基本上都会有一道问题就是数组去重,随着对于语言深入的学习,这道基...
    IloveData阅读 405评论 0 0
  • 总结一下利用JS解决去重问题的方法。总体思路: 构建一个新的空数组。 遍历数组,将不重复的元素Push到新数组中。...
    Zchao阅读 125评论 0 0
  • 数组去重 数组去重,一般需求是给你一个数组,调用去重方法,返回数值副本,副本中没有重复元素。一般来说,两个元素通过...
    我叫陈小宝T_T阅读 425评论 3 3
  • 1.简单的循环遍历 最容易想到的方法就是for循环遍历数组,并用indexOf判断是否在数组中已经存在相同的值。 ...
    April_Le阅读 349评论 0 1