JavaScipt數組去重

	// 方法1  雙層循環
	// var array = [1,1,"1","1"];
	// function unique(array) {
	// 	// res用來存儲結果
	// 	var res = [];
	// 	for(var i = 0; i < array.length; i++) {
	// 		for(var j = 0; j < res.length; j++) {
	// 			if(array[i]  === res[j]) {
	// 				break;
	// 			}
	// 		}
	// 		// 若是array[i]是唯一的,那麼執行完循環,j等於res.length
	// 		if(j === res.length) {
	// 			res.push(array[i])
	// 		}
	// 	}
	// 	return res;
	// }
	// console.log(unique(array)); //[1,"1"]
	// 在這個方法中使用循環嵌套,最外層循環array,裏面循環res,若是array[i]的值跟res[j]的值相等,就跳出循環;若是不等於就說明元素是唯一的,這時候j的值就會等於res的長度,根據這個特色進行判斷,將值添加進res


	// 方法2  indexOf -1
	// 使用indexOf能夠簡化內層循環
	// var array = [2,2,"2","2"];
	// function unique(array) {
	// 	var res = [];
	// 	for(var i = 0; i < array.length; i++) {
	// 		var current = array[i];
	// 		if(res.indexOf(current) === -1) {
	// 			res.push(current)
	// 		}
	// 	}
	// 	return res;
	// }
	// console.log(unique(array));  //[2,"2"]


	// 方法3  ES6
	// 可使用Set和Map數據結構,以Set爲例,ES6提供了新的數據結構Set,它相似於數組,可是成員的值都是惟一的,沒有重複的值
	// Set
	// var array = [3,3,"3","3"];
	// function unique(array) {
	// 	return Array.from(new Set(array));
	// }
	// console.log(unique(array));  //[3,"3"]
		// 能夠簡化爲:
		// function unique(array) {
		// 	return [...new Set(array)];
		// }
		// console.log(unique(array));  //[3,"3"]
			// 還能夠簡化爲: 
			// var unique = (a) => [...new Set(a)]

	// Map
	// var array = [3,3,"3","3"];
	// function unique(array) {
	// 	const seen = new Map()
	// 	return array.filter((a) => !seen.has(a) && seen.set(a,1))
	// }
	// console.log(unique(array));  //[3,"3"]


	// 方法4  排序後去重
	// 先將要去重的數組使用sort方法排序後,相同的值會被排在一塊兒,而後就能夠只判斷當前元素與上一個元素知否相同,相同就說明重複,不相同就添加進res
	// var array = [4,4,"4","4"];
	// function unique(array) {
	// 	var res = [];
	// 	var sortedArray = array.concat().sort();
	// 	var seen;
	// 	for(var i = 0; i < sortedArray.length; i++) {
	// 		if(!i || seen !== sortedArray[i]) {
	// 			res.push(sortedArray[i]);
	// 		}
	// 		seen = sortedArray[i];
	// 	}
	// 	return res;

	// }
	// console.log(unique(array));  //[4,"4"]
	// 若是對一個已經排好序的數組去重,這種方法的效率高於indexOf

	// 方法5  filter
	// 使用此方法來簡化外層循環
	// var array = [5,5,"5","5"];
	// function unique(array) {
	// 	var res = array.filter(function(item,index,array) {
	// 		return array.indexOf(item) === index;
	// 	})
	// 	return res;
	// }
	// console.log(unique(array));  //[5,"5"]

	// 排序去重的方法:
	// var array = [5,5,"5","5"];
	// function unique(array) {
	// 	return array.concat().sort().filter(function(item,index,array) {
	// 		return !index || item !== array[index -1]
	// 	})
	// }
	// console.log(unique(array));

	// 方法6 Object鍵值對
	// 這種方法是利用一個空的 Object 對象,咱們把數組的值存成 Object 的 key 值,好比 Object[value1] = true,在判斷另外一個值的時候,若是 Object[value2]存在的話,就說明該值是重複的。示例代碼以下:

// var array = [1, 2, 1, 1, '1'];

// function unique(array) {
//     var obj = {};
//     return array.filter(function(item, index, array){
//         return obj.hasOwnProperty(item) ? false : (obj[item] = true)
//     })
// }

// console.log(unique(array)); // [1, 2]
// 咱們能夠發現,是有問題的,由於 1 和 '1' 是不一樣的,可是這種方法會判斷爲同一個值,這是由於對象的鍵值只能是字符串,因此咱們可使用 typeof item + item 拼成字符串做爲 key 值來避免這個問題:

// var array = [1, 2, 1, 1, '1'];

// function unique(array) {
//     var obj = {};
//     return array.filter(function(item, index, array){
//         return obj.hasOwnProperty(typeof item + item) ? false : (obj[typeof item + item] = true)
//     })
// }

// console.log(unique(array)); // [1, 2, "1"]
// 然而,即使如此,咱們依然沒法正確區分出兩個對象,好比 {value: 1} 和 {value: 2},由於 typeof item + item 的結果都會是 object[object Object],不過咱們可使用 JSON.stringify 將對象序列化:

// var array = [{value: 1}, {value: 1}, {value: 2}];

// function unique(array) {
//     var obj = {};
//     return array.filter(function(item, index, array){
//         console.log(typeof item + JSON.stringify(item))
//         return obj.hasOwnProperty(typeof item + JSON.stringify(item)) ? false : (obj[typeof item + JSON.stringify(item)] = true)
//     })
// }

// console.log(unique(array)); // [{value: 1}, {value: 2}]

// 方法7 unique API
// 寫一個名爲 unique 的工具函數,根據一個參數 isSorted 判斷傳入的數組是不是已排序的,若是爲 true,就判斷相鄰元素是否相同,若是爲 false,就使用 indexOf 進行判斷

// var array1 = [1, 2, '1', 2, 1];
// var array2 = [1, 1, '1', 2, 2];

// // 初版
// function unique(array, isSorted) {
//     var res = [];
//     var seen = [];

//     for (var i = 0, len = array.length; i < len; i++) {
//         var value = array[i];
//         if (isSorted) {
//             if (!i || seen !== value) {
//                 res.push(value)
//             }
//             seen = value;
//         }
//         else if (res.indexOf(value) === -1) {
//             res.push(value);
//         }        
//     }
//     return res;
// }

// console.log(unique(array1)); // [1, 2, "1"]
// console.log(unique(array2, true)); // [1, "1", 2]
// 優化

// 新需求:字母的大小寫視爲一致,好比'a'和'A',保留一個就能夠了!

// 雖然能夠先處理數組中的全部數據,好比將全部的字母轉成小寫,而後再傳入unique函數,可是有沒有方法能夠省掉處理數組的這一遍循環,直接就在去重的循環中作呢?

// var array3 = [1, 1, 'a', 'A', 2, 2];

// // 第二版
// // iteratee 英文釋義:迭代 重複
// function unique(array, isSorted, iteratee) {
//     var res = [];
//     var seen = [];

//     for (var i = 0, len = array.length; i < len; i++) {
//         var value = array[i];
//         var computed = iteratee ? iteratee(value, i, array) : value;
//         if (isSorted) {
//             if (!i || seen !== computed) {
//                 res.push(value)
//             }
//             seen = computed;
//         }
//         else if (iteratee) {
//             if (seen.indexOf(computed) === -1) {
//                 seen.push(computed);
//                 res.push(value);
//             }
//         }
//         else if (res.indexOf(value) === -1) {
//             res.push(value);
//         }        
//     }
//     return res;
// }

// console.log(unique(array3, false, function(item){
//     return typeof item == 'string' ? item.toLowerCase() : item
// })); // [1, "a", 2]
// 在這一版也是最後一版的實現中,函數傳遞三個參數:

// array:表示要去重的數組,必填

// isSorted:表示函數傳入的數組是否已排過序,若是爲 true,將會採用更快的方法進行去重

// iteratee:傳入一個函數,能夠對每一個元素進行從新的計算,而後根據處理的結果進行去重
相關文章
相關標籤/搜索