面试官在“逗”你系列:数组去重你会几种呀?

前言

数组去重是一个老生常谈的话题,也是前端童鞋在面试时的一道高频题。本文将深入的探索数组去重的原理及实现,为各位小伙伴提供多种可以反手“调戏”面试官的解决方案。

话不多说,上去就来一梭子...

数组去重核心原理

价值100W的核心原理上来就给你了...,记得留言点赞鸭!

  1. 一般我们都会创建临时变量tmp,存储不重复的元素(以数组元素存储或对象的键来存储);
  2. 遍历待去重数组arr,依次判断tmp中是否包含该元素;
  3. 若tmp中不存在该元素,则放入;否则跳过不处理。

基本上无论什么样的实现,其核心皆是如此(判断是否已存在)。不行你就留言,咱们可以battle一下

经典去重方案一:

设置tmp为对象,对象的键存储数组元素的值,最终返回对象的所有键。

function array_unique (arr) {
  if (arr.length === 0) {
    return arr
  }
  let tmp = {}
  let len = arr.length
  for (let i = 0; i < len; i++) {
    if (tmp[arr[i]] === undefined) {
      tmp[arr[i]] = i
    }
  }
  return Object.keys(tmp)
}

// 调用数组去重
let arr = [1, 2, 3, 1, 2]
let newArr = array_unique(arr)
console.log(newArr) // ['1', '2', '3']

如果你采用这种方式来回答面试官的话,你就陷入了他在内心中早早设下的陷阱:

  1. 你这种方式能区分数字和字符串吗?能区分undefined'undefined'吗?

  2. 你现在返回的数据类型还和原有的数据类型一致吗?

带着面试官的疑问,我们来看另外一种经典去重方式。

经典去重方式二:

设置tmp为数组,数组中存储唯一的元素,最终返回tmp

function array_unique (arr) {
  let len = arr.length
  if (!len) {
    return []
  }
  let tmp = []
  for (let i = 0; i < len; i++) {
    // 判断数组arr的元素是否在数组tmp中
    if (tmp.indexOf(arr[i]) === -1) {
      tmp.push(arr[i])
    }
  }
  return tmp
}
let arr = [1, 2, 3, '1', 2, undefined, undefined, 'undefined']
let newArr = array_unique(arr)
console.log(newArr) // [1, 2, 3, '1', undefined, 'undefined']

此刻,内心是否窃喜!

But, 如果你这么考虑,又陷入了面试官的另一个陷阱:

  1. 你这方式能筛选NaN吗?

好吧,面试官最大,再考虑!

数组去重方式三:

原理还是同去重方式二,只不过我们使用ES6的includes替换indexOf方法,

includes() 方法,判断数组中是否包含某个元素,如果包含返回true,否则返回false

就是这么so easy!

function array_unique (arr) {
  let len = arr.length
  if (!len) {
    return []
  }
  let tmp = []
  for (let i = 0; i < len; i++) {
    // 判断数组arr的元素是否在数组tmp中
    if (!tmp.includes(arr[i]) {
      tmp.push(arr[i])
    }
  }
  return tmp
}
let arr = [1, 2, 3, '1', 2, undefined, undefined,  'undefined', NaN, NaN]
let newArr = array_unique(arr)
console.log(newArr) // [1, 2, 3, '1', undefined, 'undefined', NaN]

此刻,你以为就结束吗?不,不可能!

面试官的坑已经在前面等你很久了:

  1. 你的这个筛选方式能区分对象吗?如{}、{a: 1}

有没有想把自己的四十米大砍刀拿出来,neng屎面试官!(图就不配了,自己脑补吧...)
然而,什么都做不了,继续想吧...

数组去重方式四:

原理同上,我们要继续换一个判断数组是否包含某元素的方法:```findIndex``

findIndex查询数组是否包含某元素,如果存在返回元素的索引,否则返回-1。它比indexOf更加先进的地方在于能传入callback,按约定方式查询。

function array_unique (arr) {
  let len = arr.length
  if (!len) {
    return []
  }
  let tmp = []
  for (let i = 0; i < len; i++) {
    // 判断数组arr的元素是否在数组tmp中
    if (tmp.findIndex((v) => JSON.stringify(v) === JSON.stringify(arr[i])) === -1) {
      tmp.push(arr[i])
    }
  }
  return tmp
}
let arr = [1, 2, 3, '1', 2, undefined, undefined,  'undefined', NaN, NaN, {}, {}, {a: 1}, {a: 1}]
let newArr = array_unique(arr)
console.log(newArr) // [1, 2, 3, '1', undefined, 'undefined', NaN, {}, {a: 1}]

终于成功啦!来来来,可以潇洒的问面试官,“您还有问题没有?”

当然,主动挑衅面试官,是要承担风险呦,有可能会因为你眨眼的时候,先眨了右眼被挂掉了...

判断数组是否包含某元素的几种方式:

给大家列个表格,好区分几个方法的作用,

方法\是否可检测 null undefined NaN {} 备注
indexOf
includes
findIndex 需传入特定的callback

小结

数组去重这道面试题,考察的知识点还是非常多的。首先是对数组的常用方法要比较熟悉,还有其他的如NaN与NaN不相等,{}与{}不相等等知识点,以及灵活多变的思维逻辑。

当然,数组去重还有其他的多种实现方式,欢迎各位小伙伴留言交流!

后记

以上就是胡哥今天给大家分享的内容,喜欢的小伙伴记得点赞收藏呦,关注胡哥有话说,学习前端不迷路,欢迎多多留言交流...

胡哥有话说,一个有技术,有情怀的胡哥!现任京东前端攻城狮一枚。

胡哥有话说,专注于大前端技术领域,分享前端系统架构,框架实现原理,最新最高效的技术实践!

(0)

相关推荐

  • js 高阶函数reduce ——数组取交集、并集

    两个数组取交集 vs 多个数组取交集 => js reduce函数的妙用 1.reduce函数的用法及取数组交集 <script> // 值集数组 let arr1 = [1,2] ...

  • 17个你可能还不知道 JS 技巧

    17个你可能还不知道 JS 技巧 原创前端小智2020-12-14 08:38:20 本文已经过原作者 Rahul 授权翻译! 1.三元运算符 新手 let hungry = true;let eat ...

  • Typescript

    Typescript

  • es6 快速入门 系列 —— 解构

    解构 我们经常使用数组或对象存储数据,然后从中提取出相关数据信息 试图解决的问题 以前开发者为了从对象或数组中提取出特定数据并赋值给变量,编写了很多重复的代码,就像这样: function demo1 ...

  • 35.数组.选择排序

    选择排序: 第一轮: 第0个与第1个比, 如果  第0个 > 第1个  那就交换位置,第0个再与第2个比...... 第二轮: 第1个与第2比, ...................直到排序完 ...

  • 基础复习1

    基础回顾1 查找元素位置 题目描述:找出元素 item 在给定数组 arr 中的位置 输出描述: 如果数组中存在 item,则返回元素在数组中的位置,否则返回 -1 function indexOf( ...

  • 前端面试题之JavaScript

    ES6语法有哪些,分别怎么用 参考链接:http://es6.ruanyifeng.com/ new的执行过程 创建一个空对象: 将构造函数的 prototype 属性赋值给新对象的 __proto_ ...

  • JavaScript 基础四

    遍历对象的属性 for...in 语句用于对数组或者对象的属性进行循环操作. for (变量 in 对象名字) { 在此执行代码 } 这个变量是自定义 符合命名规范 但是一般我们 都写为 k 或则 k ...

  • 面试官在“逗”你系列:到底应该怎么爬楼梯?!

    直奔主题 算法题是在面试过程中考察候选人逻辑思维能力.手写代码能力的一种方式,因为有一句古话说的好:"说一千道一万,不如写段代码看一看". 今天我们就来个单刀直入,直奔主题,从一个 ...

  • 面试官在“逗”你系列:不借助第三变量交换两个变量值的方案你有几种?

    引言 在我们学习编程之初,就学习过变量的赋值操作,同时也学习了将一个变量的值赋值给另外一个变量.对于交换两个变量的值,很多童鞋都有解决方案.然鹅,对于面试官提出的不借助第三变量来交换两个变量的值,你能 ...

  • 面试官:连COUNT系列函数家族都不认识,你不能算精通Excel!

    ✎ 大家好,我是雅客. 今天,我们再次给大家整理了计数类函数的所有用法. 其中计数类函数包括COUNTA.COUNTIF.COUNTIFS.COUNT,我们都给大家整理好函数的意义和使用场景了,希望对 ...

  • 挂掉面试官系列(25匹马经典面试题的完整分析)

    今天为大家分享一道非常经典的面试题,和马有关.无论是校招,还是社招,在各大公司都出现过,我也曾经问过别人. 话不多说,直接看题. 01 25匹马的经典问题 25匹马的问题:有一个赛场上共有25匹马,赛 ...

  • 吊打面试官系列:final、finally、finalize 有什么区别?

    回复"000"获取大量电子书 final.finally.finalize 有什么区别? 这是一个初级面试题,在中高级面试中也会出现. final关键字 初级回答 final ,是 ...

  • 吊打面试官系列:说说反射的用途及实现?

    反射是什么? 反射是Java程序开发语言的特征之一,它允许动态地发现和绑定类.方法.字段,以及所有其他的由于有所产生的的元素.通过反射,能够在需要时完成创建实例.调用方法和访问字段的工作. 反射机制主 ...

  • 吊打面试官系列:说说Integer缓存范围

    回复"000"获取大量电子书 本文主要大致思路为: 不管从工作中还是面试,这篇文章都应该好好看完,本人认为是非常有用的. 案例 Integer是基本类型int的封装类.平时不管是入 ...

  • 吊打面试官系列:说说hashCode和equals方法

    首先我们需要知道hashCode方法和equals方法都是属于Object类的方法.既然属于Object中public修饰的方法,那言外之就是所有对象默认都有这两个方法,只是有时候有的对象已对这两个方 ...

  • 女面试官“品”加一笔,是什么字研究生无法回答,输给高中生

    对于面试,每个人都看得非常重要,因为这是人生中非常重要的一件大事,如果面试顺利,自己就可以去自己喜欢的公司上班,对自己未来的发展是非常有利的,但是想面试通过哪有那么简单,很多大企业招聘的岗位竞争都非常 ...