力扣热门100题之缺失的第一个正数【困难】

题目描述

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

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

示例 1:

输入:nums = 1,2,0

输出:3

示例 2:

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

输出:2

示例 3:

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

输出:1

提示:


解法1 去掉负数,排序

本来想去重 但是查重复杂度太高

js![在这里插入图片描述](https://img-blog.csdnimg.cn/500eb3c0fbf341cc8f299322ed19838e.png) 复制代码
/**
 * @param {number[]} numArr
 * @return {number}
 */
var firstMissingPositive = function(nums) {
    let numArr=[]
    nums.forEach((v)=>{
        if(v>0){
            numArr.push(v);
        }
    })
    if(numArr.length==0){
        return 1;
    }
    if(numArr.length==1&&numArr[0]>1){
        return 1;
    }
    numArr.sort((a,b)=>parseInt(a)-parseInt(b))
    let min=1;
    for(let i=0;i<numArr.length;i++){
         if(min!=numArr[i]){
            return min;
        }
        if(min==numArr[i]&&numArr[i+1]!=min){
            min++;
        }
    }
    return min;
};

执行结果:

解法2 map+自增枚举

js 复制代码
/**
 * @param {number[]} numArr
 * @return {number}
 */
var firstMissingPositive = function(nums) {
    let map=new Map();
    nums.forEach(v=>{
        if(v>0){
			map.set(v,1);//过滤掉负数和0 节约has方法的时间
        }
    });
    let num=1;
    while(true){
        if(map.has(num)){
            num++;
        }else{
            return num;
        }
    }
};

执行结果:

解法3 数组替换

将原数组有的数替换到新的数组中 使得值i 与新数组i 对应 即 ai=i;

js 复制代码
/**
 * @param {number[]} numArr
 * @return {number}
 */
var firstMissingPositive = function(nums) {
    let numArr=[]
    for(let i=0;i<nums.length;i++){
        numArr[nums[i]]=nums[i];
    }
    if(numArr[1]!=1)return 1;
    for(let i=1;i<numArr.length;i++){
        if(numArr[i]!=i){
            return i;
        }
    }
    return numArr.length;
};

执行结果:

解法4 标记法

js 复制代码
/**
 * @param {number[]} nums
 * @return {number} 核心思想 长度为n的数组,其值为(1-n) 若其值不再这个范围内 那么1-n内必有一个缺失的正数找出这个数即可
 */
var firstMissingPositive = function(nums) {
    let n=nums.length;
    if(nums.indexOf(1)==-1)return 1;
    // 把负数和0变成1
    for(let i=0;i<n;i++){
        if(nums[i]<=0||nums[i]>n){
            nums[i]=1;
        }
    }
    // 到此全为1-n内的正数
    let index=0;
    for(let i=0;i<n;i++){
        index=Math.abs(nums[i])-1
        nums[index]=-Math.abs(nums[index]);
    }
    //查找第一个负数
    for(let i=0;i<n;i++){
        if(nums[i]>0){
            return i+1;
        }
    }
    return n+1;
};

执行结果:

解法5 置换法

3, 4, -1, 1\] 为例,置换后为=\>\[1, -1, 3, 4

0, 1, 2, 2\]===\>\[ 1, 2, 0, 1

js 复制代码
/**
 * @param {number[]} nums
 * @return {number}
 */
var firstMissingPositive = function(nums) {
    let n=nums.length;
    for(let i=0;i<n;i++){
        while(nums[i] > 0 && nums[i] <= n && nums[nums[i]-1] != nums[i]){
            //把对应元素交换到对应位置
            let temp=nums[nums[i]-1];
            nums[nums[i]-1]=nums[i];
            nums[i]=temp;
        }

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

执行结果:

相关推荐
代码煮茶9 小时前
React 组件封装方法论 —— 以 Todo App 为例
javascript·react.js
任沫10 小时前
Agent之Function Call
javascript·人工智能·go
默_笙11 小时前
🛬 我让 AI 帮我写了一个打飞机游戏,结果 Canvas 把我整不会了
前端·javascript
梯度不陡11 小时前
AI 到底能不能从零写软件?ProgramBench 和 RepoZero 给出了两种答案
前端·javascript·面试
胡萝卜术13 小时前
滑动窗口最大值:从暴力到单调队列,层层优化全解析
前端·javascript·面试
kyriewen14 小时前
2026 年了,这 6 个 npm 包可以卸载了——浏览器原生 API 已经能替代
前端·javascript·npm
铁皮饭盒15 小时前
bun直接tsx,优雅!
javascript·后端
_柳青杨17 小时前
一文吃透 Node.js 事件循环:从原理到 Node 20+ 重大变更
javascript·后端
JieE2121 天前
LeetCode 101. 对称二叉树|JS 递归 + 迭代双解法,彻底搞懂镜像判断
javascript·算法
冬奇Lab1 天前
AI Workflow 定义的四次演进:从 Markdown 到 JS 脚本,再到分布式多 Agent
javascript·人工智能·agent