力扣热门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 对应 即 a[i]=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;
};

执行结果:

相关推荐
清汤饺子3 分钟前
OpenClaw 本地部署教程 - 从 0 到 1 跑通你的第一只龙虾
前端·javascript·vibecoding
颜酱21 分钟前
图的数据结构:从「多叉树」到存储与遍历
javascript·后端·算法
橙某人4 小时前
LogicFlow 小地图性能优化:从「实时克隆」到「占位缩略块」!🚀
前端·javascript·vue.js
boooooooom5 小时前
讲清 Proxy + effect + track/trigger 流程
javascript·vue.js·面试
leafyyuki5 小时前
在 Vue 项目中玩转 FullCalendar:从零搭建可交互的事件日历
前端·javascript·vue.js
豆苗学前端5 小时前
彻底讲透浏览器缓存机制,吊打面试官
前端·javascript·面试
zone77396 小时前
006:RAG 入门-面试官问你,RAG 为什么要切块?
后端·算法·面试
swipe6 小时前
箭头函数与 this 面试题深度解析:从原理到实战
前端·javascript·面试
进击的尘埃7 小时前
拖拽搭建场景下的智能布局算法:栅格吸附、参考线与响应式出码
javascript
小猪努力学前端8 小时前
基于PixiJS的试玩广告开发-续篇
前端·javascript·游戏