話說面試常會碰到面試官會問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