LeetCode hoot 100 -- 缺失的第一个正整数

本题取自LeetCode hoot 100 题号41 缺失的第一个正整数

一 题目概述

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

例:nums=[1,3,4],没有出现的最小正整数为 2

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

二 思路解析

要找到没有出现的最小正整数,正常的思路应该是先排序再找出未出现的元素。然而在此我们可以利用map<int,bool>结构,先遍历nums数组,记录元素出现情况,再从map中找出第一个没有被标记为bool的元素。代码如下:

然而可以看到,题目要求使用常数级别额外空间。既然无法利用map省去排序过程,我们只能原地进行排序。遍历所有元素,并将其放到自己该待的位置 swap(nums[i], nums[nums[i]-1]);

三 代码解析

map:

cpp 复制代码
class Solution {
public:
    int firstMissingPositive(vector<int>& nums) {
        unordered_map<int, bool> mp;
        // 遍历nums,记录正整数的出现情况
        for (int x : nums) {
            if (x > 0) mp[x] = true;
        }
        // 从1开始遍历,找到第一个未被标记为true的元素,即为答案
        int i = 1;
        while (mp.count(i)) {
            i++;
        }
        return i;
    }
};

swap:

cpp 复制代码
class Solution {
public:
    int firstMissingPositive(vector<int>& nums) {
        int n = nums.size();
        
        // 把每个数放到正确位置
        for (int i = 0; i < n; ++i) {
            // 只有数字在 1~n 范围内,且不在正确位置时才交换
            // 注,此处需要循环,而不只是if判断,因为如果向后置换往后不会遍历到被置换的元素,其也                  无法回到自己正确的位置
            while (nums[i] > 0 && nums[i] <= n && nums[nums[i]-1] != nums[i]) {
                swap(nums[i], nums[nums[i]-1]);
            }
        }
        
        // 第二步:找第一个不满足 nums[i] == i+1 的位置
        for (int i = 0; i < n; ++i) {
            if (nums[i] != i + 1) {
                return i + 1;
            }
        }

        // 全部都正确,返回 n+1
        return n + 1;
    }
};

感谢阅读

相关推荐
飞Link4 小时前
深度解析孪生网络(Siamese Network):从原理、技巧到实战应用
算法·数据挖掘·回归
测试狗科研平台4 小时前
洞悉微观电荷流动,VASP计算电荷密度分布
算法·云计算·开源软件
fqq34 小时前
java基础面试题目
面试·职场和发展
Orz_Sponge_Bob4 小时前
温州市第三届青少年程序设计竞赛(小学组)题解
算法
小江的记录本5 小时前
【Java并发编程】锁机制:volatile:JMM内存模型、可见性/禁止指令重排、内存屏障、单例模式中的应用(附《思维导图》+《面试高频考点清单》)
java·后端·python·mysql·单例模式·面试·职场和发展
Noushiki5 小时前
常见的排序算法
算法·排序算法
gumichef5 小时前
二叉树链式结构的实现
算法·链表·二叉树·队列
战南诚5 小时前
力扣 之 198.打家劫舍
python·算法·leetcode
AllData公司负责人5 小时前
亲测丝滑,体验跃迁|AllData通过集成开源项目StreamPark,实时流任务调度更省心!
java·大数据·数据库·人工智能·算法·实时计算·实时开发平台