JS實(shí)現(xiàn)數(shù)組去重方法整理

前言

我們先來看下面的例子嚼贡,當(dāng)然來源與網(wǎng)絡(luò)捉片,地址《刪除數(shù)組中多個(gè)不連續(xù)的數(shù)組元素的正確姿勢(shì)

我們現(xiàn)在將數(shù)組中所有的‘ a’ 元素刪除:

var arr = ['a', 'a', 'b', 'c', 'd', 'a', 'a', 'e', 'g', 'a', 'f'];

arr.forEach(function(value, index) {

value === 'a' ? arr.splice(index, 1) : '';

})

console.log(arr);

//["a", "b", "c", "d", "a", "e", "g", "f"]

只要相鄰的‘ a’ 元素秩霍, 都沒被刪除齐婴, splice不但可以刪除元素本身茄厘, 還同時(shí)可以減少數(shù)組長度( 就是抹去一切痕跡)矮冬,

這樣導(dǎo)致后續(xù)的數(shù)組元素會(huì)代替已經(jīng)刪除的元素的位置, 但是循環(huán)是按照數(shù)組的下標(biāo)按順序刪除次哈, 這樣就會(huì)漏掉遷移的元素胎署。

看到網(wǎng)上有網(wǎng)友在說使用delete進(jìn)行操作,如下:

vararr = ['a', 'a', 'b', 'c', 'd', 'a', 'a', 'e', 'g', 'a', 'f'];

arr.forEach(function(value, index) {

? value === 'a' ?deletearr[index] : '';

})

console.log(arr); //[2: "b", 3: "c", 4: "d", 7: "e", 8: "g", 10: "f"]

但是得到的arr其實(shí)是一個(gè)非常規(guī)的數(shù)組了窑滞,也就是說其實(shí)delete主要是用于對(duì)對(duì)象屬性的操作琼牧。這確實(shí)要根據(jù)自己的需求來了。

當(dāng)然簡(jiǎn)單的實(shí)現(xiàn)如下:

vararr = ['a', 'a', 'b', 'c', 'd', 'a', 'a', 'e', 'g', 'a', 'f'];varnewArr = arr.filter(function(key) {

? ? returnkey !== 'a'})

console.log(newArr); //["b", "c", "d", "e", "g", "f"]

?下面總結(jié)下常用實(shí)現(xiàn)方式哀卫。

性能測(cè)試模板

let arr1 = Array.from(newArray(100000),(item,index) => {

? ? return index;

})

let arr2 = Array.from(newArray(50000),(item,index) => {

? ? returnindex + index;

})function distinct(a,b) {

? ? // 數(shù)組去重}

console.time('去重計(jì)算時(shí)間');

console.log('去重后的長度', distinct(arr1, arr2).length);

console.timeEnd('去重計(jì)算時(shí)間');

上面創(chuàng)建了兩個(gè)數(shù)組長度巨坊,1個(gè)10W,1個(gè)5W長度的數(shù)組此改,合并到一個(gè)數(shù)組趾撵,然后進(jìn)行去重,驗(yàn)證下去重的計(jì)算時(shí)間是多長共啃。

方法一:Array.filter() + indexOf

function distinct(a,b) {

? ? let arr = a.concat(b);

? ? returnarr.filter((item,index) => {

? ? ? ? returnarr.indexOf(item) === index;

? ? })

}

思路就是ES6 中的 Array.filter() 遍歷數(shù)組占调,并結(jié)合 indexOf 來排除重復(fù)項(xiàng)∫萍簦看下結(jié)果:

從截圖可以看出究珊,計(jì)算時(shí)間花費(fèi)了19753ms的時(shí)間。

方法二:使用 for...of + includes()

function distinct(a,b) {

? ? let arr = a.concat(b)

? ? ? ? let result = []

? ? ? ? for (let i of arr) {

? ? ? ? ? ? !result.includes(i) && result.push(i)

? ? ? ? }

? ? ? ? return result

}

這種方式跟Array.filter() + indexOf實(shí)現(xiàn)思路差不多挂滓,結(jié)果如圖所示:

從圖上可以看出苦银,計(jì)算的時(shí)間跟Array.filter() + indexOf的時(shí)間差不多啸胧。

方法三:雙重 for 循環(huán)

function distinct(a, b) {

? ? let arr = a.concat(b);

? ? for(let i=0, len=arr.length; i

? ? ? ? for(let j=i+1; j

? ? ? ? ? ? if(arr[i] == arr[j]) {

? ? ? ? ? ? ? ? arr.splice(j, 1);

? ? ? ? ? ? ? ? // splice 會(huì)改變數(shù)組長度,所以要將數(shù)組長度 len 和下標(biāo) j 減一len--;

? ? ? ? ? ? ? ? j--;

? ? ? ? ? ? }

? ? ? ? }

? ? }

? ? return arr

}

優(yōu)點(diǎn):簡(jiǎn)單易懂

缺點(diǎn):占用內(nèi)存高幔虏,速度慢

看下結(jié)果:

方法四:Array.sort()

首先使用 sort() 將數(shù)組進(jìn)行排序

然后比較相鄰元素是否相等纺念,從而排除重復(fù)項(xiàng)。


function distinct(a, b) {

? ? let arr = a.concat(b)

? ? arr.sort(function(a,b){//對(duì)數(shù)組進(jìn)行排序才能方便比較returna - b;

? ? })

? ? let result = [arr[0]]

? ? for(let i=1, len=arr.length; i

? ? ? ? arr[i] !== arr[i-1] && result.push(arr[i])

? ? }

? ? return result

}


從上面代碼可以看出來想括,做了一次排序和一次循環(huán)陷谱,所以效率比前面都高,結(jié)果如圖所示:

可以看出瑟蜈,結(jié)果只花了255ms左右時(shí)間烟逊。

方法五:for...of + Object

利用對(duì)象的屬性不能相同的特點(diǎn)進(jìn)行去重,代碼如下:

function distinct(a, b) {

? ? let arr = a.concat(b)

? ? let result = []

? ? let obj = {}

? ? for (let i of arr) {

? ? ? ? if(!obj[i]) {

? ? ? ? ? ? result.push(i)

? ? ? ? ? ? obj[i] = 1? ? ? ? }

? ? }

? ? return result

}

執(zhí)行結(jié)果如圖所示:

結(jié)果只花費(fèi)了43ms左右時(shí)間铺根。

方法六:ES6的new Set()

function distinct(a, b) {

? ? returnArray.from(new Set([...a, ...b]))

}

Set數(shù)據(jù)結(jié)構(gòu)宪躯,它類似于數(shù)組,其成員的值都是唯一的位迂。

利用Array.from將Set結(jié)構(gòu)轉(zhuǎn)換成數(shù)組访雪。

結(jié)果如圖所示:

從結(jié)果看出來該方法執(zhí)行只花了151ms左右的時(shí)間。

總結(jié)

從上面幾種方法得出如下結(jié)論:

數(shù)組去重要么使用for...of + Object方式掂林,要么使用ES6的 new Set()方式臣缀。

從執(zhí)行結(jié)果看for...of + Object的效率應(yīng)該是最高的(只在當(dāng)前量級(jí)的計(jì)算結(jié)果來看)。

附錄

附錄中添加瀏覽器Array對(duì)象支持indexOf和forEach的polyfill:

Array.prototype.indexOf = Array.prototype.indexOf ||function(item) {

? ? for(vari = 0, j =this.length; i < j; i++) {

? ? ? ? if(this[i] === item) {

? ? ? ? ? ? return i;

? ? ? ? }

? ? }

? ? return-1;

}

Array.prototype.forEach = Array.prototype.forEach ||function(callback, thisArg) {

? ? if(!callback ||typeofcallback !== 'function')return;

? ? for(vari = 0, j =this.length; i < j; i++) {

? ? ? ? callback.call(thisArg, this[i], i,this);

? ? }

}

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末泻帮,一起剝皮案震驚了整個(gè)濱河市精置,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌锣杂,老刑警劉巖脂倦,帶你破解...
    沈念sama閱讀 222,000評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異元莫,居然都是意外死亡狼讨,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,745評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門柒竞,熙熙樓的掌柜王于貴愁眉苦臉地迎上來政供,“玉大人,你說我怎么就攤上這事朽基〔几簦” “怎么了?”我有些...
    開封第一講書人閱讀 168,561評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵稼虎,是天一觀的道長衅檀。 經(jīng)常有香客問我,道長霎俩,這世上最難降的妖魔是什么哀军? 我笑而不...
    開封第一講書人閱讀 59,782評(píng)論 1 298
  • 正文 為了忘掉前任沉眶,我火速辦了婚禮,結(jié)果婚禮上杉适,老公的妹妹穿的比我還像新娘谎倔。我一直安慰自己,他們只是感情好猿推,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,798評(píng)論 6 397
  • 文/花漫 我一把揭開白布片习。 她就那樣靜靜地躺著,像睡著了一般蹬叭。 火紅的嫁衣襯著肌膚如雪藕咏。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,394評(píng)論 1 310
  • 那天秽五,我揣著相機(jī)與錄音孽查,去河邊找鬼。 笑死坦喘,一個(gè)胖子當(dāng)著我的面吹牛卦碾,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播起宽,決...
    沈念sama閱讀 40,952評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼济榨!你這毒婦竟也來了坯沪?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,852評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤擒滑,失蹤者是張志新(化名)和其女友劉穎腐晾,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體丐一,經(jīng)...
    沈念sama閱讀 46,409評(píng)論 1 318
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡藻糖,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,483評(píng)論 3 341
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了库车。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片巨柒。...
    茶點(diǎn)故事閱讀 40,615評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖柠衍,靈堂內(nèi)的尸體忽然破棺而出洋满,到底是詐尸還是另有隱情,我是刑警寧澤珍坊,帶...
    沈念sama閱讀 36,303評(píng)論 5 350
  • 正文 年R本政府宣布牺勾,位于F島的核電站,受9級(jí)特大地震影響阵漏,放射性物質(zhì)發(fā)生泄漏驻民。R本人自食惡果不足惜翻具,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,979評(píng)論 3 334
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望回还。 院中可真熱鬧裆泳,春花似錦、人聲如沸懦趋。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,470評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽仅叫。三九已至帜篇,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間诫咱,已是汗流浹背笙隙。 一陣腳步聲響...
    開封第一講書人閱讀 33,571評(píng)論 1 272
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留坎缭,地道東北人竟痰。 一個(gè)月前我還...
    沈念sama閱讀 49,041評(píng)論 3 377
  • 正文 我出身青樓,卻偏偏與公主長得像掏呼,于是被迫代替她去往敵國和親坏快。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,630評(píng)論 2 359

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

  • 一憎夷、測(cè)試模版 數(shù)組去重是一個(gè)老生常談的問題莽鸿,網(wǎng)上流傳著有各種各樣的解法 為了測(cè)試這些解法的性能,我寫了一個(gè)測(cè)試模版...
    過期貓糧_45f1閱讀 874評(píng)論 0 0
  • 方法一: 雙層循環(huán)拾给,外層循環(huán)元素祥得,內(nèi)層循環(huán)時(shí)比較值 如果有相同的值則跳過,不相同則push進(jìn)數(shù)組 Array.pr...
    小旭同志閱讀 866評(píng)論 0 0
  • js數(shù)組常用方法整理 標(biāo)簽:js 數(shù)組 前言:這篇文章主要是一些歸納總結(jié)蒋得,大部分參考的是MDN的程序级及,也加了一些自...
    wrma閱讀 720評(píng)論 0 10
  • Javascript有很多數(shù)組的方法,有的人有W3C的API额衙,還可以去MDN上去找饮焦,但是我覺得API上說的不全,M...
    頑皮的雪狐七七閱讀 4,095評(píng)論 0 6
  • 關(guān)于數(shù)組長度: 如果為一個(gè)數(shù)組元素賦值窍侧,它的索引 i 大于或等于現(xiàn)有數(shù)組的長度時(shí)追驴,length的值會(huì)自動(dòng)設(shè)置為 i...
    栗子酥小小閱讀 518評(píng)論 0 0