【力扣hot100题】(115)缺失的第一个正数

想法是用一个数组记录所有数字是否出现,最后遍历这个数组。

cpp 复制代码
class Solution {
public:
    int firstMissingPositive(vector<int>& nums) {
        bool num[100001];
        memset(num,0,sizeof(num));
        for(int i=0;i<nums.size();i++){
            if(nums[i]<100001&&nums[i]>=0) num[nums[i]]=1;
        }
        for(int i=1;i<100001;i++) if(num[i]!=1) return i;
        return 100001;
    }
};

空间复杂度是O(100001),何尝不算是一种O(1)(?)

答案的两种方法都很巧妙。一种是上面这种的优化,即用原本的数组记录该数是否存在:

cpp 复制代码
class Solution {
public:
    int firstMissingPositive(vector<int>& nums) {
        int n=nums.size();
        for(int i=0;i<n;i++){
            if(nums[i]<=0) nums[i]=n+1;
        }
        for(int i=0;i<n;i++){
            if(abs(nums[i])<=n) nums[abs(nums[i])-1]=-abs(nums[abs(nums[i])-1]);
        }
        for(int i=0;i<n;i++){
            if(nums[i]>0) return i+1;
        }
        return n+1;
    }
};

若当前数(的绝对值)在1,n范围内,就将数对应于数组的位置变为负数(相当于记录该数存在,并不动这个位置原本的数),为了避免和原本的负数混淆,一开始就将非正数记为超过n的数。

还有一种方法是置换,若当前数在1,n内就将当前数和对应位置的数交换:

cpp 复制代码
class Solution {
public:
    int firstMissingPositive(vector<int>& nums) {
        for(int i=0;i<nums.size();i++){
            while(nums[i]<nums.size()&&nums[i]>0&&nums[i]!=nums[nums[i]-1]) swap(nums[i],nums[nums[i]-1]);
        }
        for(int i=0;i<nums.size();i++){
            if(nums[i]!=i+1) return i+1;
        }
        return nums.size()+1;
    }
};
相关推荐
通信小呆呆9 小时前
当算法有了“五感”:多模态数据融合如何向人体感官协同学习?
人工智能·学习·算法·机器学习·机器人
benben0449 小时前
强化学习之DQN算法族(基于gymnasium开发)
算法
小小工匠10 小时前
Redis - 事务机制:能实现 ACID 属性吗
数据结构·redis·性能优化·并发·持久化
玖玥拾10 小时前
C/C++ 数据结构(七)栈、容器适配器
c语言·数据结构·c++··容器适配器
何以解忧,唯有..11 小时前
Go语言循环语句详解:for、range与循环控制
开发语言·算法·golang
想吃火锅100511 小时前
【leetcode】88.合并两个有序数组js
算法
生成论实验室12 小时前
机器人:一个自主运动的系统
人工智能·算法·语言模型·机器人·自动驾驶·agi·安全架构
Qres82112 小时前
算法复键——树状数组
数据结构·算法
H1785350909612 小时前
SolidWorks第四部分_直接实体建模特征9_替换面原理
线性代数·算法·机器学习·3d建模·solidworks