javascript數(shù)組去重3種方法的性能測試與比較
更新時間:2013年03月26日 11:40:44 作者:
面試題中有一題數(shù)組去重,首先想到的是對象存鍵值的方法可是遇到不同類型又能轉(zhuǎn)換成同樣的字符串的就完了接下來為大家介紹下雙重循環(huán)/存鍵值和類型實(shí)現(xiàn)去重,感興趣的各位可以參考下哈
昨天參加的一個前端面試,其中有一題數(shù)組去重,首先想到的是對象存鍵值的方法,代碼如下
方法一:(簡單存鍵值)
Array.prototype.distinct1 = function() {
var i=0,tmp={},that=this.slice(0)
this.length=0;
for(;i<that.length;i++){
if(!(that[i] in tmp)){
this[this.length]=that[i];
tmp[that[i]]=true;
}
}
return this;
};
上面的方法不復(fù)雜,思路也簡單,可是遇到不同類型又能轉(zhuǎn)換成同樣的字符串的就完了,比如1和"1";于是又用了傳統(tǒng)的雙重循環(huán),代碼如下
方法二:(雙重循環(huán))
Array.prototype.distinct2 = function() {
var i=0,flag,that=this.slice(0);
this.length=0;
for(;i<that.length;i++){
var tmp=that[i];
flag=true;
for(var j=0;j<this.length;j++){
if(this[j]===tmp){flag=false;break}
}
if(flag)this[this.length]=tmp;
}
return this;
};
上面這種方法得到了想要的結(jié)果,但是兩層循環(huán)效率比較低,我們再想辦法從第一種方法上上手,然后加上用字符串來保存數(shù)組項(xiàng)的類型,有新類型就連接字符串加上去,查找的時候就發(fā)現(xiàn)一個有保存的類型就把存的類型的字符串替換為空,代碼如下
方法三:(存鍵值和類型)
Array.prototype.distinct4 = function() {
var i=0,tmp={},t2,that=this.slice(0),one;
this.length=0;
for(;i<that.length;i++){
one=that[i];
t2=typeof one;
if(!(one in tmp)){
this[this.length]=one;
tmp[one]=t2;
}else if(tmp[one].indexOf(t2)==-1){
this[this.length]=one;
tmp[one]+=t2;
}
}
return this;
};
為了區(qū)別下不同數(shù)據(jù)的各種算法的效率差距,取幾種極端的例子來驗(yàn)證下,首先看看1-80全部數(shù)組項(xiàng)都不一樣循環(huán)1000次的情況,好吧,IE6弱爆了
下面是80項(xiàng)全部重復(fù)的循環(huán)1000次的情況,綜合上面的數(shù)據(jù)一起發(fā)現(xiàn)除了IE6-8其它瀏覽器的雙重循環(huán)表現(xiàn)都不錯,而IE6-8雙重循環(huán)要慢10-20倍左右,悲催啊。如果你的網(wǎng)站只支持IE9以上的就可以放心用雙循環(huán)的方法了,否則還是使用健值的方法,根據(jù)數(shù)據(jù)的情況選擇使用方法一或方法三(圖中的方法四,才發(fā)現(xiàn)來不及改圖,原來的方法三是用了Array的indexOf,因?yàn)樗俣嚷也患嫒菥蜎]放出來了)
方法一:(簡單存鍵值)
復(fù)制代碼 代碼如下:
Array.prototype.distinct1 = function() {
var i=0,tmp={},that=this.slice(0)
this.length=0;
for(;i<that.length;i++){
if(!(that[i] in tmp)){
this[this.length]=that[i];
tmp[that[i]]=true;
}
}
return this;
};
上面的方法不復(fù)雜,思路也簡單,可是遇到不同類型又能轉(zhuǎn)換成同樣的字符串的就完了,比如1和"1";于是又用了傳統(tǒng)的雙重循環(huán),代碼如下
方法二:(雙重循環(huán))
復(fù)制代碼 代碼如下:
Array.prototype.distinct2 = function() {
var i=0,flag,that=this.slice(0);
this.length=0;
for(;i<that.length;i++){
var tmp=that[i];
flag=true;
for(var j=0;j<this.length;j++){
if(this[j]===tmp){flag=false;break}
}
if(flag)this[this.length]=tmp;
}
return this;
};
上面這種方法得到了想要的結(jié)果,但是兩層循環(huán)效率比較低,我們再想辦法從第一種方法上上手,然后加上用字符串來保存數(shù)組項(xiàng)的類型,有新類型就連接字符串加上去,查找的時候就發(fā)現(xiàn)一個有保存的類型就把存的類型的字符串替換為空,代碼如下
方法三:(存鍵值和類型)
復(fù)制代碼 代碼如下:
Array.prototype.distinct4 = function() {
var i=0,tmp={},t2,that=this.slice(0),one;
this.length=0;
for(;i<that.length;i++){
one=that[i];
t2=typeof one;
if(!(one in tmp)){
this[this.length]=one;
tmp[one]=t2;
}else if(tmp[one].indexOf(t2)==-1){
this[this.length]=one;
tmp[one]+=t2;
}
}
return this;
};
為了區(qū)別下不同數(shù)據(jù)的各種算法的效率差距,取幾種極端的例子來驗(yàn)證下,首先看看1-80全部數(shù)組項(xiàng)都不一樣循環(huán)1000次的情況,好吧,IE6弱爆了
IE9:Chrome:
Firefox: IE6:
下面是80項(xiàng)全部重復(fù)的循環(huán)1000次的情況,綜合上面的數(shù)據(jù)一起發(fā)現(xiàn)除了IE6-8其它瀏覽器的雙重循環(huán)表現(xiàn)都不錯,而IE6-8雙重循環(huán)要慢10-20倍左右,悲催啊。如果你的網(wǎng)站只支持IE9以上的就可以放心用雙循環(huán)的方法了,否則還是使用健值的方法,根據(jù)數(shù)據(jù)的情況選擇使用方法一或方法三(圖中的方法四,才發(fā)現(xiàn)來不及改圖,原來的方法三是用了Array的indexOf,因?yàn)樗俣嚷也患嫒菥蜎]放出來了)
IE9: Chrome:
Firefox: IE6:
您可能感興趣的文章:
- js數(shù)組去重的常用方法總結(jié)
- JS實(shí)現(xiàn)數(shù)組去重方法總結(jié)(六種方法)
- js數(shù)組去重的5種算法實(shí)現(xiàn)
- js取兩個數(shù)組的交集|差集|并集|補(bǔ)集|去重示例代碼
- 史上最全JavaScript數(shù)組去重的十種方法(推薦)
- JavaScript數(shù)組去重的兩種方法推薦
- js實(shí)現(xiàn)數(shù)組去重、判斷數(shù)組以及對象中的內(nèi)容是否相同
- js算法中的排序、數(shù)組去重詳細(xì)概述
- JavaScript數(shù)組去重的五種方法
- JS實(shí)現(xiàn)數(shù)組簡單去重及數(shù)組根據(jù)對象中的元素去重操作示例
- JavaScript 高性能數(shù)組去重的方法
- 高性能js數(shù)組去重(12種方法,史上最全)
相關(guān)文章
ES6?Promise.all的使用方法以及其細(xì)節(jié)詳解
Promise對象用于表示一個異步操作的最終完成(或失敗)及其結(jié)果值,下面這篇文章主要給大家介紹了關(guān)于ES6?Promise.all的使用方法以及其細(xì)節(jié)的相關(guān)資料,需要的朋友可以參考下2022-07-07ES6中Promise的使用方法實(shí)例總結(jié)
這篇文章主要介紹了ES6中Promise的使用方法,結(jié)合實(shí)例形式總結(jié)分析了Promise對象中的各種常用方法及基本使用技巧,需要的朋友可以參考下2020-02-02解析微信JS-SDK配置授權(quán),實(shí)現(xiàn)分享接口
這篇文章是微信開發(fā)的很多前置條件,包括了服務(wù)端基于JAVA的獲取和緩存全局的access_token,獲取和緩存全局的jsapi_ticket,以及前端配置授權(quán)組件封裝,調(diào)用分享組件封裝。需要的朋友可以看下2016-12-12