JS 数组去重
作者:沉静地闪光
数组去重,一般都是在面试的时候才会碰到,一般是要求手写数组去重方法的代码。如果是被提问到,数组去重的方法有哪些?你能答出其中的10种,面试官很有可能对你刮目相看。
在真实的项目中碰到的数组去重,一般都是后台去处理,很少让前端处理数组去重。虽然日常项目用到的概率比较低,但还是需要了解一下,以防面试的时候可能回被问到。注:写的匆忙,加上这几天有点忙,还没有非常认真核对过,不过思路是没有问题,可能一些小细节出错而已。
数组去重的方法
一、利用ES6 Set去重(ES6中最常用)
1 2 3 4 5 6 | function unique (arr) { return Array.from( new Set(arr)) } var arr = [1,1, 'true' , 'true' , true , true ,15,15, false , false , undefined,undefined, null , null , NaN, NaN, 'NaN' , 0, 0, 'a' , 'a' ,{},{}]; console.log(unique(arr)) //[1, "true", true, 15, false, undefined, null, NaN, "NaN", 0, "a", {}, {}] |
不考虑兼容性,这种去重的方法代码最少。这种方法还无法去掉“{}”空对象,后面的高阶方法会添加去掉重复“{}”的方法。
二、利用for嵌套for,然后splice去重(ES5中最常用)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 | function unique(arr){ for ( var i=0; i<arr.length; i++){ for ( var j=i+1; j<arr.length; j++){ if (arr[i]==arr[j]){ //第一个等同于第二个,splice方法删除第二个 arr.splice(j,1); j--; } } } return arr; } var arr = [1,1, 'true' , 'true' , true , true ,15,15, false , false , undefined,undefined, null , null , NaN, NaN, 'NaN' , 0, 0, 'a' , 'a' ,{},{}]; console.log(unique(arr)) //[1, "true", 15, false, undefined, NaN, NaN, "NaN", "a", {…}, {…}] //NaN和{}没有去重,两个null直接消失了 |
双层循环,外层循环元素,内层循环时比较值。值相同时,则删去这个值。
想快速学习更多常用的ES6语法,可以看我之前的文章《学习ES6笔记──工作中常用到的ES6语法》。
三、利用indexOf去重
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 | function unique(arr) { if (!Array.isArray(arr)) { console.log( 'type error!' ) return } var array = []; for ( var i = 0; i < arr.length; i++) { if (array .indexOf(arr[i]) === -1) { array .push(arr[i]) } } return array; } var arr = [1,1, 'true' , 'true' , true , true ,15,15, false , false , undefined,undefined, null , null , NaN, NaN, 'NaN' , 0, 0, 'a' , 'a' ,{},{}]; console.log(unique(arr)) // [1, "true", true, 15, false, undefined, null, NaN, NaN, "NaN", 0, "a", {…}, {…}] //NaN、{}没有去重 |
新建一个空的结果数组,for 循环原数组,判断结果数组是否存在当前元素,如果有相同的值则跳过,不相同则push进数组。
四、利用sort()
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 | function unique(arr) { if (!Array.isArray(arr)) { console.log( 'type error!' ) return ; } arr = arr.sort() var arrry= [arr[0]]; for ( var i = 1; i < arr.length; i++) { if (arr[i] !== arr[i-1]) { arrry.push(arr[i]); } } return arrry; } var arr = [1,1, 'true' , 'true' , true , true ,15,15, false , false , undefined,undefined, null , null , NaN, NaN, 'NaN' , 0, 0, 'a' , 'a' ,{},{}]; console.log(unique(arr)) // [0, 1, 15, "NaN", NaN, NaN, {…}, {…}, "a", false, null, true, "true", undefined] //NaN、{}没有去重 |
利用sort()排序方法,然后根据排序后的结果进行遍历及相邻元素比对。
五、利用对象的属性不能相同的特点进行去重(这种数组去重的方法有问题,不建议用,有待改进)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 | function unique(arr) { if (!Array.isArray(arr)) { console.log( 'type error!' ) return } var arrry= []; var obj = {}; for ( var i = 0; i < arr.length; i++) { if (!obj[arr[i]]) { arrry.push(arr[i]) obj[arr[i]] = 1 } else { obj[arr[i]]++ } } return arrry; } var arr = [1,1, 'true' , 'true' , true , true ,15,15, false , false , undefined,undefined, null , null , NaN, NaN, 'NaN' , 0, 0, 'a' , 'a' ,{},{}]; console.log(unique(arr)) //[1, "true", 15, false, undefined, null, NaN, 0, "a", {…}] //两个true直接去掉了,NaN和{}去重 |
六、利用includes
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 | function unique(arr) { if (!Array.isArray(arr)) { console.log( 'type error!' ) return } var array =[]; for ( var i = 0; i < arr.length; i++) { if ( !array.includes( arr[i]) ) { //includes 检测数组是否有某个值 array.push(arr[i]); } } return array } var arr = [1,1, 'true' , 'true' , true , true ,15,15, false , false , undefined,undefined, null , null , NaN, NaN, 'NaN' , 0, 0, 'a' , 'a' ,{},{}]; console.log(unique(arr)) //[1, "true", true, 15, false, undefined, null, NaN, "NaN", 0, "a", {…}, {…}] //{}没有去重 |
七、利用hasOwnProperty
1 2 3 4 5 6 7 8 9 | function unique(arr) { var obj = {}; return arr.filter( function (item, index, arr){ return obj.hasOwnProperty( typeof item + item) ? false : (obj[ typeof item + item] = true ) }) } var arr = [1,1, 'true' , 'true' , true , true ,15,15, false , false , undefined,undefined, null , null , NaN, NaN, 'NaN' , 0, 0, 'a' , 'a' ,{},{}]; console.log(unique(arr)) //[1, "true", true, 15, false, undefined, null, NaN, "NaN", 0, "a", {…}] //所有的都去重了 |
利用hasOwnProperty 判断是否存在对象属性
八、利用filter
1 2 3 4 5 6 7 8 9 | function unique(arr) { return arr.filter( function (item, index, arr) { //当前元素,在原始数组中的第一个索引==当前索引值,否则返回当前元素 return arr.indexOf(item, 0) === index; }); } var arr = [1,1, 'true' , 'true' , true , true ,15,15, false , false , undefined,undefined, null , null , NaN, NaN, 'NaN' , 0, 0, 'a' , 'a' ,{},{}]; console.log(unique(arr)) //[1, "true", true, 15, false, undefined, null, "NaN", 0, "a", {…}, {…}] |
九、利用递归去重
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 | function unique(arr) { var array= arr; var len = array.length; array.sort( function (a,b){ //排序后更加方便去重 return a - b; }) function loop(index){ if (index >= 1){ if (array[index] === array[index-1]){ array.splice(index,1); } loop(index - 1); //递归loop,然后数组去重 } } loop(len-1); return array; } var arr = [1,1, 'true' , 'true' , true , true ,15,15, false , false , undefined,undefined, null , null , NaN, NaN, 'NaN' , 0, 0, 'a' , 'a' ,{},{}]; console.log(unique(arr)) //[1, "a", "true", true, 15, false, 1, {…}, null, NaN, NaN, "NaN", 0, "a", {…}, undefined] |
十、利用Map数据结构去重
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 | function arrayNonRepeatfy(arr) { let map = new Map(); let array = new Array(); // 数组用于返回结果 for ( let i = 0; i < arr.length; i++) { if (map .has(arr[i])) { // 如果有该key值 map .set(arr[i], true ); } else { map .set(arr[i], false ); // 如果没有该key值 array .push(arr[i]); } } return array ; } var arr = [1,1, 'true' , 'true' , true , true ,15,15, false , false , undefined,undefined, null , null , NaN, NaN, 'NaN' , 0, 0, 'a' , 'a' ,{},{}]; console.log(unique(arr)) //[1, "a", "true", true, 15, false, 1, {…}, null, NaN, NaN, "NaN", 0, "a", {…}, undefined] |
创建一个空Map数据结构,遍历需要去重的数组,把数组的每一个元素作为key存到Map中。由于Map中不会出现相同的key值,所以最终得到的就是去重后的结果。
十一、利用reduce+includes
1 2 3 4 5 6 | function unique(arr){ return arr.reduce((prev,cur) => prev.includes(cur) ? prev : [...prev,cur],[]); } var arr = [1,1, 'true' , 'true' , true , true ,15,15, false , false , undefined,undefined, null , null , NaN, NaN, 'NaN' , 0, 0, 'a' , 'a' ,{},{}]; console.log(unique(arr)); // [1, "true", true, 15, false, undefined, null, NaN, "NaN", 0, "a", {…}, {…}] |
十二、[...new Set(arr)]
[...new Set(arr)]
//代码就是这么少----(其实,严格来说并不算是一种,相对于第一种方法来说只是简化了代码)
PS:有些文章提到了foreach+indexOf数组去重的方法,个人觉得都是大同小异,所以没有写上去。
来自
https://segmentfault.com/a/1190000016418021
热评:
yhw:
总结下来楼主这12中无非就两类:
- 两层循环法
- 利用语法自身键不可重复性
- 其他的都是用了语言不通的api,算不上单独的一种方法,比如indexOf 和includes 两层循环 都是同一类
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 按钮权限的设计及实现
· 25岁的心里话