【LeetCode热题100】41. 缺失的第一个正数(数组)

一.题目要求

给你一个未排序的整数数组 nums ,请你找出其中没有出现的最小的正整数。

请你实现时间复杂度为 O(n) 并且只使用常数级别额外空间的解决方案。

二.题目难度

困难

三.输入样例

示例 1:

输入:nums = [1,2,0]

输出:3

解释:范围 [1,2] 中的数字都在数组中。

示例 2:

输入:nums = [3,4,-1,1]

输出:2

解释:1 在数组中,但 2 没有。

示例 3:

输入:nums = [7,8,9,11,12]

输出:1

解释:最小的正数 1 没有出现。

提示:

1 <= nums.length <= 105

-231 <= nums[i] <= 231 - 1

四.解题思路

原地哈希

思路:长度为n的数组nums缺失的第一个正数的范围只可能是1 ~ n+1

极端情况:长度为5 数组nums为 1, 2, 3, 4, 5 即前五个数都出现,那么缺失的数就是n + 1 = 6

遍历一遍数组,将nums[i]的值映射到原本属于它的位置上,此题的映射规则为 对于数 val 映射到 nums[val - 1]

如nums[5] = 4; 此时值4处在了值6应该在的位置,要把4映射到nums[3]

以此类推,把所有符合1 - n的数都映射到其对应的位置,那么在nums[i]中,某个数不是i+1,则这个数就是所求缺失值

否则表面所有数都是按照1,2,3,4的顺序映射 那么不存在的最小整数就是n + 1

五.代码实现

cpp 复制代码
class Solution {
public:
    int firstMissingPositive(vector<int>& nums) {

        int n = nums.size();

        for (int i = 0; i < n; i++) {
            // 注意:
            // 这里while的目的是让nums[i]里的元素被最终定下来,最后留在nums[i]的元素一定要么是i + 1要么是一个不符合1 ~ n的数
            while (nums[i] > 0 && nums[i] <= n && nums[i] != i + 1 &&
                   nums[nums[i] - 1] != nums[i]) {
                std::swap(nums[nums[i] - 1], nums[i]);
            }
        }

        for (int i = 0; i < n; i++) {
            if (nums[i] != i + 1)
                return i + 1;
        }

        return n + 1;
    }
};

六.题目总结

原地哈希技巧的应用

相关推荐
Asmalin6 小时前
【代码随想录day 35】 力扣 01背包问题 一维
算法·leetcode·职场和发展
剪一朵云爱着6 小时前
力扣2779. 数组的最大美丽值
算法·leetcode·排序算法
tao3556677 小时前
【Python刷力扣hot100】283. Move Zeroes
开发语言·python·leetcode
未知陨落8 小时前
LeetCode:98.颜色分类
算法·leetcode
Haooog11 小时前
98.验证二叉搜索树(二叉树算法题)
java·数据结构·算法·leetcode·二叉树
Young_Zn_Cu13 小时前
LeetCode刷题记录(持续更新中)
算法·leetcode
天选之女wow13 小时前
【代码随想录算法训练营——Day31】贪心算法——56.合并区间、738.单调递增的数字、968.监控二叉树
算法·leetcode·贪心算法
Miraitowa_cheems16 小时前
LeetCode算法日记 - Day 64: 岛屿的最大面积、被围绕的区域
java·算法·leetcode·决策树·职场和发展·深度优先·推荐算法
_不会dp不改名_17 小时前
leetcode_1382 将二叉搜索树变平衡树
算法·leetcode·职场和发展
Q741_14719 小时前
C++ 位运算 高频面试考点 力扣 面试题 17.19. 消失的两个数字 题解 每日一题
c++·算法·leetcode·面试·位运算