LeetCode - 448 - 找出所有数组中消失的数字(find-numbers-disappeared-in-array)

Create by jsliang on 2019-7-30 07:50:532019-7-30 08:36:11

一 目录

不折腾的前端,和咸鱼有什么区别

| 目录 | | --- | | 一 目录 | | 二 前言 | | 三 解题 | |  3.1 解法 - Map | |  3.2 解法 - Set |

二 前言

  • 难度:简单

  • 涉及知识:数组

  • 题目地址:https://leetcode-cn.com/problems/find-all-numbers-disappeared-in-an-array/

  • 题目内容

给定一个范围在  1 ≤ a[i] ≤ n ( n = 数组大小 ) 的 整型数组,	
数组中的元素一些出现了两次,另一些只出现一次。	
找到所有在 [1, n] 范围之间没有出现在数组中的数字。	
您能在不使用额外空间且时间复杂度为O(n)的情况下完成这个任务吗? 	
你可以假定返回的数组不算在额外空间内。	
示例:	
输入:	
[4,3,2,7,8,2,3,1]	
输出:	
[5,6]

三 解题

小伙伴可以先自己在本地尝试解题,再回来看看 jsliang 的解题思路。

3.1 解法 - Map

  • 解题代码

const findDisappearedNumbers = (nums) => {	
  let map = new Map();	
  for (let i = 1; i <= nums.length; i++) {	
    map.set(i, 1);	
  }	
  for (let j = 0; j < nums.length; j++) {	
    if (map.get(nums[j])) {	
      map.delete(nums[j]);	
    }	
  }	
  let result = [];	
  for (let key of map.keys()) {	
    result.push(key);	
  }	
  return result;	
};
  • 执行测试

  1. nums: [4,3,2,7,8,2,3,1]

  2. return

[5, 6]
  • LeetCode Submit

√ Accepted	
  √ 34/34 cases passed (192 ms)	
  √ Your runtime beats 48.21 % of javascript submissions	
  √ Your memory usage beats 5.17 % of javascript submissions (61.9 MB)
  • 知识点

  1. Map:保存键值对。任何值(对象或者原始值) 都可以作为一个键或一个值。 Map 详细介绍

  2. push(): push() 方法将一个或多个元素添加到数组的末尾,并返回该数组的新长度。 push() 详细介绍

  • 解题思路

犹豫即将败北

首先,拿到题目第一眼,想想要不要尝试下使用 Map

所以直接上手:

const findDisappearedNumbers = (nums) => {	
  let map = new Map();	
  for (let i = 1; i <= nums.length; i++) {	
    map.set(i, 1);	
  }	
  for (let j = 0; j < nums.length; j++) {	
    if (map.get(nums[j])) {	
      map.delete(nums[j]);	
    }	
  }	
  let result = [];	
  for (let key of map.keys()) {	
    result.push(key);	
  }	
  return result;	
};

然后,Submit 提交看看:

√ Accepted	
  √ 34/34 cases passed (192 ms)	
  √ Your runtime beats 48.21 % of javascript submissions	
  √ Your memory usage beats 5.17 % of javascript submissions (61.9 MB)

哟嚯,居然可行。

最后,我们讲讲使用 Map 的简单思路:

步骤 1:设置一个 1-n 的哈希表:

let map = new Map();	
for (let i = 1; i <= nums.length; i++) {	
  map.set(i, 1);	
}	
// Map {	
//   1 => 1,	
//   2 => 1	
//   3 => 1,	
//   4 => 1,	
//   5 => 1,	
//   6 => 1,	
//   7 => 1,	
//   8 => 1,	
// }

步骤 2:遍历原数组,删除对应哈希表中出现的数字:

for (let j = 0; j < nums.length; j++) {	
  if (map.get(nums[j])) {	
    map.delete(nums[j]);	
  }	
}	
// Map {	
//   5 => 1,	
//   6 => 1,	
// }

步骤 3:遍历哈希表,将其转换为数组:

let result = [];	
for (let key of map.keys()) {	
  result.push(key);	
}	
return result;	
// [ 5, 6 ]

这样,就完成了这道题的破解。

不得不承认这是简单而又繁杂的破解思路,我们应该进行更多尝试。

3.2 解法 - Set

  • 解题代码

const findDisappearedNumbers = (nums) => {	
  let newNums = [...new Set(nums)].sort((a, b) => a - b);	
  let result = [];	
  for (let i = 0; i < nums.length; i++) {	
    if (newNums[i] != (i + 1)) {	
      newNums.splice(i, 0, (i + 1));	
      result.push((i + 1));	
    }	
  }	
  return result;	
};
  • 执行测试

  1. nums: [4,3,2,7,8,2,3,1]

  2. return

[5, 6]
  • LeetCode Submit

√ Accepted	
  √ 34/34 cases passed (1236 ms)	
  √ Your runtime beats 21.12 % of javascript submissions	
  √ Your memory usage beats 5.17 % of javascript submissions (51.9 MB)
  • 知识点

  1. Set: Set 对象允许你存储任何类型的唯一值,无论是原始值或者是对象引用。 Set 详细介绍

  2. sort():排序,保持返回数组的数字为顺序排列。 sort() 详细介绍

  3. splice(): splice() 方法通过删除或替换现有元素或者原地添加新的元素来修改数组,并以数组形式返回被修改的内容。此方法会改变原数组。 splice() 详细介绍

  4. push(): push() 方法将一个或多个元素添加到数组的末尾,并返回该数组的新长度。 push() 详细介绍

  • 解题思路

首先,在上文中,我们讲到了使用 Map 进行求解。而讲起 Mapjsliang 又想到了它的死对头 Set,然后尝试了一波:

const findDisappearedNumbers = (nums) => {	
  let newNums = [...new Set(nums)].sort((a, b) => a - b);	
  let result = [];	
  for (let i = 0; i < nums.length; i++) {	
    if (newNums[i] != (i + 1)) {	
      newNums.splice(i, 0, (i + 1));	
      result.push((i + 1));	
    }	
  }	
  return result;	
};

然后,我们看看它的思路:

  1. 通过 Set 和 sort() 去重 + 排序。

  2. 遍历数组,注意我们遍历的是 nums 的长度,毕竟求解的范围是 1-n。

  3. 如果我们去重后的 newNums 不存在值,那么我们给它补充一个值,同时,补充的这个值,就是我们的求解值之一。

最后,我们将 result 返回出去即可。

  • 进一步思考

那么,在上面,我们使用了自己的智慧进行解题,下面该看看大佬们的解题技巧啦:

解法 1:

var findDisappearedNumbers = function(nums) {	
  var newNums = [];	
  var result = [];	
  for (var i = 1; i <= nums.length; i++) {	
    newNums[i] = nums.findIndex(e => e === i);	
  }	
  for (var i in newNums) {	
    if (newNums[i] === -1) {	
      result.push(i);	
    }	
  }	
  return result;	
};

Submit:

√ Accepted	
  √ 34/34 cases passed (5108 ms)	
  √ Your runtime beats 18.73 % of javascript submissions	
  √ Your memory usage beats 5.17 % of javascript submissions (52.2 MB)

解法 2:

const findDisappearedNumbers = nums => {	
  const arr = [...nums];	
  for (let i = 0; i < nums.length; i++) {	
    arr[nums[i] - 1] > 0 && (arr[nums[i] - 1] *= -1);	
  }	
  return arr.map((item, index) => item > 0 && index + 1).filter(Number);	
};

Submit:

√ Accepted	
  √ 34/34 cases passed (160 ms)	
  √ Your runtime beats 71.31 % of javascript submissions	
  √ Your memory usage beats 29.31 % of javascript submissions (45.8 MB)

解法 3:

var findDisappearedNumbers = function(nums) {	
  let ans = Array(nums.length)	
    .fill(null)	
    .map((_, h) => h + 1);	
  for (let i = 0; i < nums.length; i++) {	
    if (ans[nums[i] - 1] == nums[i]) {	
      ans[nums[i] - 1] = 0;	
    }	
  }	
  ans = ans.filter(value => {	
    return value != 0;	
  });	
  return ans;	
};

Submit:

√ Accepted	
  √ 34/34 cases passed (144 ms)	
  √ Your runtime beats 88.45 % of javascript submissions	
  √ Your memory usage beats 29.31 % of javascript submissions (45.2 MB)

非常精彩~

那么该题的解题就此结束啦!


不折腾的前端,和咸鱼有什么区别!

640?wx_fmt=jpeg

jsliang 会每天更新一道 LeetCode 题解,从而帮助小伙伴们夯实原生 JS 基础,了解与学习算法与数据结构。

扫描上方二维码,关注 jsliang 的公众号,让我们一起折腾!

640?wx_fmt=png

jsliang 的文档库 由 梁峻荣 采用 知识共享 署名-非商业性使用-相同方式共享 4.0 国际 许可协议进行许可。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值