力扣热门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;
};

执行结果:

相关推荐
SpongeG1 小时前
数据结构第三周做题总结_链表
数据结构·算法·链表
everyStudy1 小时前
前端五种排序
前端·算法·排序算法
小珑也要变强1 小时前
队列基础概念
c语言·开发语言·数据结构·物联网
醉竺1 小时前
【高阶数据结构】二叉搜索树的插入、删除和查找(精美图解+完整代码)
数据结构
little redcap3 小时前
第十九次CCF计算机软件能力认证-乔乔和牛牛逛超市
数据结构·c++·算法
muyierfly3 小时前
34.贪心算法1
算法·贪心算法
她似晚风般温柔7894 小时前
Uniapp + Vue3 + Vite +Uview + Pinia 分商家实现购物车功能(最新附源码保姆级)
开发语言·javascript·uni-app
Jiaberrr5 小时前
前端实战:使用JS和Canvas实现运算图形验证码(uniapp、微信小程序同样可用)
前端·javascript·vue.js·微信小程序·uni-app
everyStudy6 小时前
JS中判断字符串中是否包含指定字符
开发语言·前端·javascript
luthane6 小时前
python 实现average mean平均数算法
开发语言·python·算法