【JavaScript】數(shù)組去重

話說面試常會碰到面試官會問JavaScript實現(xiàn)數(shù)組去重的問題,最近剛好在學(xué)習(xí)有關(guān)于JavaScript數(shù)組相關(guān)的知識,趁此機會整理了一些有關(guān)于JavaScript數(shù)組去重的方法搁痛。

下面這些數(shù)組去重的方法是自己收集和整理的泞歉,如有不對希望指正文中不對之處逼侦。

雙重循環(huán)去重


這個方法使用了兩個for循環(huán)做遍歷匿辩。整個思路是:

  • 構(gòu)建一個空數(shù)組用來存放去重后的數(shù)組
  • 外面的for循環(huán)對原數(shù)組做遍歷,每次從數(shù)組中取出一個元素與結(jié)果數(shù)組做對比
  • 如果原數(shù)組取出的元素與結(jié)果數(shù)組元素相同榛丢,則跳出循環(huán);反之則將其存放到結(jié)果數(shù)組中

代碼如下:

Array.prototype.unique1 = function () { 
 // 構(gòu)建一個新數(shù)組铲球,存放結(jié)果 var newArray = [this[0]]; 
 // for循環(huán),每次從原數(shù)組中取出一個元素 
 // 用取出的元素循環(huán)與結(jié)果數(shù)組對比 
 for (var i = 1; i < this.length; i++) { 
  var repeat = false; 
  for (var j=0; j < newArray.length; j++) { 
  // 原數(shù)組取出的元素與結(jié)果數(shù)組元素相同 
   if(this[i] == newArray[j]) { 
    repeat = true; break; 
   } 
  } 
  if(!repeat) { 
  // 如果結(jié)果數(shù)組中沒有該元素晰赞,則存放到結(jié)果數(shù)組中     
   newArray.push(this[i]);
  } 
 } 
 return newArray;
}

假設(shè)我們有一個這樣的數(shù)組:

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]

據(jù)說這種方法比較耗時稼病,費性能。簡單做個測試(測試方法寫得比較拙逼):

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('雙重循環(huán)去重方法使用時間是:' + (tEnd - tStart) + 'ms'); 
 return re;
}
test();

在Chrome控制器運行上面的代碼掖鱼,測試雙重循環(huán)去重所費時間:11031ms然走。

上面的方法可以使用forEach()方法和indexOf()方法模擬實現(xiàn):

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

通過unique1.apply(arr)或unique1.call(arr)調(diào)用。不過這種方法效率要快得多戏挡,同樣的上面測試代碼芍瑞,所費時間5423ms,幾乎快了一半褐墅。

排序遍歷去重


先使用sort()方法對原數(shù)組做一個排序拆檬,排完序之后對數(shù)組做遍歷,并且檢查數(shù)組中的第i個元素與結(jié)果數(shù)組中最后一個元素是否相同妥凳。如果不同竟贯,則將元素放到結(jié)果數(shù)組中。

Array.prototype.unique2 = function () { 
 // 原數(shù)組先排序 
 this.sort(); 
 // 構(gòu)建一個新數(shù)組存放結(jié)果 
 var newArray = []; 
 for (var i = 1; i < this.length; i++) { 
  // 檢查原數(shù)中的第i個元素與結(jié)果中的最后一個元素是否相同 
  // 因為排序了猾封,所以重復(fù)元素會在相鄰位置 
  if(this[i] !== newArray[newArray.length - 1]) { 
   // 如果不同澄耍,將元素放到結(jié)果數(shù)組中 
   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"]

這種方法有兩個特色:
去重后的數(shù)組會做排序,主要是因為原數(shù)在去重前做了排序去重后的數(shù)組晌缘,與數(shù)字相同的數(shù)字字符無法區(qū)分齐莲,比如'1'和1

使用同樣的方法,測試所費時間:1232ms磷箕。

對象鍵值對法


這種去重方法實現(xiàn)思路是:
創(chuàng)建一個JavaScript對象以及新數(shù)組使用for循環(huán)遍歷原數(shù)組选酗,每次取出一個元素與JavaScript對象的鍵做對比
如果不包含,將存入對象的元素的值推入到結(jié)果數(shù)組中,并且將存入object
對象中該屬性名的值設(shè)置為1

代碼如下:

Array.prototype.unique3 = function () { 
 // 構(gòu)建一個新數(shù)組存放結(jié)果 
 var newArray = []; 
 // 創(chuàng)建一個空對象 
 var object = {}; 
 // for循環(huán)時岳枷,每次取出一個元素與對象進(jìn)行對比 
 // 如果這個元素不重復(fù)芒填,則將它存放到結(jié)果數(shù)中 
 // 同時把這個元素的內(nèi)容作為對象的一個屬性,并賦值為1, 
 // 存入到第2步建立的對象中 
 for (var i = 0; i < this.length; i++){ 
  // 檢測在object對象中是否包含遍歷到的元素的值 
  if(!object[typeof(this[i]) + this[i]]) { 
  // 如果不包含空繁,將存入對象的元素的值推入到結(jié)果數(shù)組中
   newArray.push(this[i]); 
   // 如果不包含殿衰,存入object對象中該屬性名的值設(shè)置為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"]

同樣的,不同的鍵可能會被誤認(rèn)為一樣盛泡;例如: a[1]闷祥、a["1"]。這種方法所費時間:621ms傲诵。 這種方法所費時間是最短凯砍,但就是占用內(nèi)存大一些箱硕。

除了上面幾種方法,還有其他幾種方法如下:

// 方法四
Array.prototype.unique4 = function () { 
 // 構(gòu)建一個新數(shù)組存放結(jié)果 
 var newArray = []; 
 // 遍歷整個數(shù)組 
 for (var i = 0; i < this.length; i++) { 
  // 遍歷是否有重復(fù)的值 
  for (j = i + 1; j < this.length; j++) { 
   // 如果有相同元素悟衩,自增i變量剧罩,跳出i的循環(huán) 
   if(this[i] === this[j]) { 
    j = ++i; 
   }  
  } 
  // 如果沒有相同元素,將元素推入到結(jié)果數(shù)組中  
  newArray.push(this[i]); 
 } 
 return newArray;
}

Chrome測試結(jié)果

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'無法區(qū)分惠昔。

// 方法五
Array.prototype.unique5 = function () { 
 // 構(gòu)建一個新數(shù)組存放結(jié)果 
 var newArray = []; 
 // 遍歷整個數(shù)組 
 for (var i = 0; i < this.length; i++) { 
  // 如果當(dāng)前數(shù)組的第i值保存到臨時數(shù)組,那么跳過 
  var index = this[i]; 
  // 如果數(shù)組項不在結(jié)果數(shù)組中钳榨,將這個值推入結(jié)果數(shù)組中 
  if (newArray.indexOf(index) === -1) { 
   newArray.push(index); 
  } 
 } 
 return newArray;
}

Chrome測試結(jié)果:

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'無法區(qū)分纽门。所費時間:14361ms薛耻。

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

測試結(jié)果如下:

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;
}

測試結(jié)果:

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赏陵。
方法雖然很多種饼齿,但相比下來,下面這種方法是較為優(yōu)秀的方案:

//方法三
Array.prototype.unique3 = function () { 
 // 構(gòu)建一個新數(shù)組存放結(jié)果 
 var newArray = []; 
 // 創(chuàng)建一個空對象 
 var object = {}; 
 // for循環(huán)時蝙搔,每次取出一個元素與對象進(jìn)行對比 
 // 如果這個元素不重復(fù)缕溉,則將它存放到結(jié)果數(shù)中 
 // 同時把這個元素的內(nèi)容作為對象的一個屬性,并賦值為1, 
 // 存入到第2步建立的對象中 
 for (var i = 0; i < this.length; i++){ 
  // 檢測在object對象中是否包含遍歷到的元素的值 
  if(!object[typeof(this[i]) + this[i]]) { 
  // 如果不包含吃型,將存入對象的元素的值推入到結(jié)果數(shù)組中
   newArray.push(this[i]); 
   // 如果不包含证鸥,存入object對象中該屬性名的值設(shè)置為1 
   object[typeof(this[i]) + this[i]] = 1; 
  } 
 } 
 return newArray;
}

但在ES6去重還有更簡單,更優(yōu)化的方案勤晚,比如:

// 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))
}

轉(zhuǎn)載自:http://www.w3cplus.com/javascript/remove-duplicates-from-javascript-array.html

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末枉层,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子赐写,更是在濱河造成了極大的恐慌鸟蜡,老刑警劉巖,帶你破解...
    沈念sama閱讀 207,113評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件挺邀,死亡現(xiàn)場離奇詭異揉忘,居然都是意外死亡,警方通過查閱死者的電腦和手機端铛,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,644評論 2 381
  • 文/潘曉璐 我一進(jìn)店門泣矛,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人禾蚕,你說我怎么就攤上這事您朽。” “怎么了夕膀?”我有些...
    開封第一講書人閱讀 153,340評論 0 344
  • 文/不壞的土叔 我叫張陵虚倒,是天一觀的道長美侦。 經(jīng)常有香客問我,道長魂奥,這世上最難降的妖魔是什么菠剩? 我笑而不...
    開封第一講書人閱讀 55,449評論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮耻煤,結(jié)果婚禮上具壮,老公的妹妹穿的比我還像新娘。我一直安慰自己哈蝇,他們只是感情好棺妓,可當(dāng)我...
    茶點故事閱讀 64,445評論 5 374
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著炮赦,像睡著了一般怜跑。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上吠勘,一...
    開封第一講書人閱讀 49,166評論 1 284
  • 那天性芬,我揣著相機與錄音,去河邊找鬼剧防。 笑死植锉,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的峭拘。 我是一名探鬼主播俊庇,決...
    沈念sama閱讀 38,442評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼鸡挠!你這毒婦竟也來了辉饱?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,105評論 0 261
  • 序言:老撾萬榮一對情侶失蹤宵凌,失蹤者是張志新(化名)和其女友劉穎鞋囊,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體瞎惫,經(jīng)...
    沈念sama閱讀 43,601評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡溜腐,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,066評論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了瓜喇。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片挺益。...
    茶點故事閱讀 38,161評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖乘寒,靈堂內(nèi)的尸體忽然破棺而出望众,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 33,792評論 4 323
  • 正文 年R本政府宣布烂翰,位于F島的核電站夯缺,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏甘耿。R本人自食惡果不足惜踊兜,卻給世界環(huán)境...
    茶點故事閱讀 39,351評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望佳恬。 院中可真熱鬧捏境,春花似錦、人聲如沸毁葱。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,352評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽倾剿。三九已至筷频,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間柱告,已是汗流浹背截驮。 一陣腳步聲響...
    開封第一講書人閱讀 31,584評論 1 261
  • 我被黑心中介騙來泰國打工笑陈, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留际度,地道東北人。 一個月前我還...
    沈念sama閱讀 45,618評論 2 355
  • 正文 我出身青樓涵妥,卻偏偏與公主長得像乖菱,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子蓬网,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,916評論 2 344

推薦閱讀更多精彩內(nèi)容

  • Why underscore (覺得這部分眼熟的可以直接跳到下一段了...) 最近開始看 underscore.j...
    韓子遲閱讀 1,379評論 0 57
  • 偶然和同事談到面試的Javascript問題窒所,其中基本上都會有一道問題就是數(shù)組去重,隨著對于語言深入的學(xué)習(xí)帆锋,這道基...
    IloveData閱讀 426評論 0 0
  • 總結(jié)一下利用JS解決去重問題的方法吵取。總體思路: 構(gòu)建一個新的空數(shù)組锯厢。 遍歷數(shù)組皮官,將不重復(fù)的元素Push到新數(shù)組中。...
    Zchao閱讀 127評論 0 0
  • 數(shù)組去重 數(shù)組去重实辑,一般需求是給你一個數(shù)組捺氢,調(diào)用去重方法,返回數(shù)值副本剪撬,副本中沒有重復(fù)元素摄乒。一般來說,兩個元素通過...
    我叫陳小寶T_T閱讀 438評論 3 3
  • 1.簡單的循環(huán)遍歷 最容易想到的方法就是for循環(huán)遍歷數(shù)組,并用indexOf判斷是否在數(shù)組中已經(jīng)存在相同的值馍佑。 ...
    April_Le閱讀 363評論 0 1