力扣最热一百题——缺失的第一个正数

news2024/11/13 15:15:35

目录

题目链接:41. 缺失的第一个正数 - 力扣(LeetCode)

题目描述

示例

提示:

解法一:标记数组法

1. 将非正数和超出范围的数替换

2. 使用数组下标标记存在的数字

3. 找到第一个未标记的位置

4. 为什么时间复杂度是 O(n)?

5. 常数空间?

Java写法:

运行时间

C++写法:

运行时间

时间复杂度以及空间复杂度 

解法二:交换至正确的位置

1. 将每个数放到正确的位置上

2. 查找第一个未按顺序排列的位置

3. 如果所有数字都按顺序排列

为什么时间复杂度是 O(n)?

为什么空间复杂度是 O(1)?

困惑为什么交换的时候是while而不是if

Java解法:

运行时间

C++解法:

运行时间

时间复杂度以及空间复杂度

总结


题目链接:41. 缺失的第一个正数 - 力扣(LeetCode)

注:下述题目描述和示例均来自力扣

题目描述

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

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

示例

示例 1:

输入:nums = [1,2,0]
输出:3
解释:范围 [1,2] 中的数字都在数组中。

示例 2:

输入:nums = [3,4,-1,1]
输出:2
解释:1 在数组中,但 2 没有。

示例 3:

输入:nums = [7,8,9,11,12]
输出:1
解释:最小的正数 1 没有出现。

提示:

  • 1 <= nums.length <= 10^5
  • -2^31 <= nums[i] <= 2^31 - 1


解法一:标记数组法

1. 将非正数和超出范围的数替换

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

        首先,代码遍历数组 nums,将其中所有非正数(即小于等于0的数)或大于 n 的数(n 是数组的长度)替换为 n + 1。因为我们只关心数组中出现的正整数,且最小的正整数应该在1到 n 的范围内,所以将这些不相关的数(非正数和大于 n 的数)统一设置为 n + 1(一个无效的值,确保不会干扰后续的逻辑)。

2. 使用数组下标标记存在的数字

 
for (int i = 0; i < n; ++i) {
    int num = Math.abs(nums[i]);
    if (num <= n) {
        nums[num - 1] = -Math.abs(nums[num - 1]);
    }
}

这一步的目标是通过数组中的数字来标记哪些正整数是存在的。具体逻辑是:

  • 对每个数 num = abs(nums[i]),如果 num 在1到 n 的范围内,则将 nums[num-1] 的值设为负数。这相当于利用下标 num-1 来记录数字 num 是否出现过。
  • 如果某个数字 num 出现了,就将位置 num-1 上的数字设为负数,表示该位置已经被标记。

3. 找到第一个未标记的位置

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

在这一步,代码再次遍历数组,查找第一个值为正数的下标 i,表示 i+1 这个数字没有出现过。因为在第二步中,所有出现过的数字的对应位置已经被标记为负数,所以第一个正数的位置就是缺失的最小正整数。

4. 为什么时间复杂度是 O(n)?

  • 每个元素最多被处理两次:第一次是在将非正数替换为 n+1 时,第二次是在通过下标标记数字时。因此,总体的遍历次数是线性的,即 O(n)。

5. 常数空间?

  • 除了输入数组 nums 本身外,代码没有使用额外的数据结构(比如数组、栈、队列等)。空间复杂度是 O(1),因为对数组的修改都是就地进行的。

Java写法:

class Solution {
    public int firstMissingPositive(int[] nums) {
        // 取得数组的长度
        int n = nums.length;
        // 由于负数并不在我们的考虑范围里面,所以全部放到涉及不到的地方N+1
        for (int i = 0; i < n; i++) {
            if (nums[i] <= 0) {
                nums[i] = n + 1;
            }
        }
        // 将每个数都打上“标记”
        for (int i = 0; i < n; i++) {
            // 由于这里的数可能在打标记的过程中被修改为负数
            // 所以在这里再取值的时候要取为绝对值
            int num = Math.abs(nums[i]);
            if (num <= n) {
                // 采用绝对值的负数,防止被打两次负数变成正数
                nums[num - 1] = -Math.abs(nums[num - 1]);
            }
        }
        // 找有没有是正数的,有的话就是他的位置
        for (int i = 0; i < n; i++) {
            if (nums[i] > 0) {
                return i + 1;
            }
        }
        // 刚才没有找到正数,那么就是数组长度加一的位置了
        return n + 1;
    }
}

运行时间

 

C++写法:

class Solution {
public:
    int firstMissingPositive(vector<int>& nums) {
        // 由于在java那里的注释我已经写的很详细了,这里我就随便写写了
        // 获取数组的长度
        int n = nums.size();

        // 将小于等于零的数(非正整数)都设置为无关紧要的位置也就是n+1
        for(int i = 0; i < n; i++){
            if(nums[i] <= 0){
                nums[i] = n + 1;
            }
        }

        // 打标记,将在[1,n+1]中的数,其大小对应的下标-1上的数置为负数
        for(int i = 0; i < n; i++){
            int num = abs(nums[i]);
            if(num <= n){
                // 为了防止nums[num - 1]已经被标记(取为负数)这里取绝对值
                nums[num - 1] = -abs(nums[num - 1]);
            }
        }

        // 找有没有正数
        for(int i = 0; i < n; i++){
            if(nums[i] > 0){
                // 找到了
                return i + 1;
            }
        }

        // 没找到
        return n + 1;

    }
};

运行时间

时间复杂度以及空间复杂度 




解法二:交换至正确的位置

1. 将每个数放到正确的位置上

 
for (int i = 0; i < n; ++i) {
    while (nums[i] > 0 && nums[i] <= n && nums[nums[i] - 1] != nums[i]) {
        int temp = nums[nums[i] - 1];
        nums[nums[i] - 1] = nums[i];
        nums[i] = temp;
    }
}

这一部分的核心思想是将数组中的数字放到正确的位置上。每个数字 nums[i],如果它在1到 n 的范围内,那么它应该出现在数组的第 nums[i] - 1 个位置。

  • 通过 while 循环,代码不断检查 nums[i] 是否满足以下条件:
    • nums[i] 是正数,并且在1到 n 的范围内。
    • nums[i] 还没有被放到它应该在的位置上(即 nums[nums[i] - 1] != nums[i])。
  • 如果条件满足,就将 nums[i] 与它应该在的位置 nums[nums[i] - 1] 进行交换,直到每个数都被放到了正确的位置上,或者 nums[i] 已经不需要再交换了。

这个过程类似于 "桶排序"(Cyclic Sort)的思想,把数组看作一个映射,通过交换将每个数字放到对应的桶(即数组位置)中。

2. 查找第一个未按顺序排列的位置

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

在第二部分,代码再次遍历数组,寻找第一个下标 i,使得 nums[i] != i + 1,即第 i+1 这个数字没有出现在数组中。如果找到这样的位置,就说明 i + 1 是第一个缺失的最小正整数。

3. 如果所有数字都按顺序排列

return n + 1;

如果所有数字都已经按顺序排列了,那么数组中的数从 1n 都出现过,这时缺失的最小正整数是 n + 1

为什么时间复杂度是 O(n)?

  • 每个元素最多会被交换一次,因为每次交换都把元素放到其正确的位置上。交换的次数是有限的,因此整个过程的时间复杂度是 O(n)。

为什么空间复杂度是 O(1)?

  • 除了输入数组本身外,代码没有使用额外的数据结构,交换操作是就地进行的,因此空间复杂度为 O(1)。

困惑为什么交换的时候是while而不是if

  • 多个元素错位的情况
    在这个问题中,目标是将每个元素放到它的正确位置,例如数字 k 应该放在数组的第 k-1 位置上。由于数组是未排序的,一个元素在经过一次交换后,它可能仍然没有被放到正确的位置。因此,代码需要反复检查并继续交换,直到这个元素被放置在正确的位置上。

  • 确保每个元素到达正确的位置
    使用 while 的目的是让每个元素继续交换,直到它符合条件为止,即 nums[i] > 0 && nums[i] <= n && nums[nums[i] - 1] == nums[i]。如果使用 if,只能在当前条件下交换一次,但在某些情况下,交换一次后新的 nums[i] 可能仍然需要继续交换。例如,如果两个错位的元素在第一次交换后,新的元素也不在正确位置上,则需要再次交换。

  • 示例: 假设数组是 [3, 4, -1, 1],我们希望让每个元素放在正确的位置:

    • 第一个元素 3 应该放在位置 2(即下标 3 - 1 = 2)。
    • 交换之后数组变为:[-1, 4, 3, 1]。注意此时 nums[0] 变为了 -1
    • 但是第一个元素现在是 -1,不符合条件(nums[0] > 0 && nums[0] <= n),所以停止处理这个位置。
    • 接着处理第二个元素 44 应该在位置 3,交换后数组变为:[-1, 1, 3, 4]。此时 nums[1] = 1 也不在正确的位置,需要再一次交换,把 1 放到位置 0
    • 通过 while 循环,1 被正确放置在位置 0,最终得到数组 [1, -1, 3, 4]

    如果这里使用的是 if 而不是 while,那么在某些情况下,数组中的元素可能没有被交换到最终的正确位置,需要额外的遍历或逻辑来完成任务。


Java解法:

class Solution {
    public int firstMissingPositive(int[] nums) {
        // 先获取数组的长度
        int len = nums.length;
        // 进入交换数组的逻辑
        for(int i = 0; i < len; i++){
            // 由于nums[i](本来的值)和nums[nums[i] - 1](应该在的位置的值)
            // 可能在交换之后不在正确的位置上,所以需要一直交换,直到在正确的位置上
            while(nums[i] >= 1 && nums[i] <= len && nums[i] != nums[nums[i] - 1]){
                int temp = nums[i];
                nums[i] = nums[nums[i] - 1];
                nums[temp - 1] = temp;
            }
        }

        // 找自己数值跟位置对不上的
        for(int i = 0; i < len; i++){
            if(nums[i] != i + 1){
                return i + 1;
            }
        }

        // 那就是最后一位了
        return len + 1;
    }
}

运行时间

C++解法:

class Solution {
public:
    int firstMissingPositive(vector<int>& nums) {
        // 由于在java那里的注释我已经写的很详细了,这里我就随便写写了
        // 获取数组的长度
        int len = nums.size();
        
        // 交换逻辑,确保数字放到正确的位置
        for (int i = 0; i < len; i++) {
            // 不断交换,直到当前的nums[i]是有效值并且放在正确的位置上
            while (nums[i] >= 1 && nums[i] <= len && nums[i] != nums[nums[i] - 1]) {
                // 交换时也需要防止nums[i]的值被覆盖后产生的错误
                int temp = nums[i];
                nums[i] = nums[temp - 1];
                nums[temp - 1] = temp;
            }
        }

        // 检查第一个位置不正确的元素
        for (int i = 0; i < len; i++) {
            if (nums[i] != i + 1) {
                return i + 1;
            }
        }

        // 如果所有数字都按顺序排列,则缺少的是len + 1
        return len + 1;
    }
};

运行时间



时间复杂度以及空间复杂度


总结

        哇塞,不愧是力扣的困难题目,我现在真的越来越喜欢写困难的题目了,每次写完虽然可能要花点时间,但是每次写完都是一次来自大脑的洗涤,思维的活跃,这种逆天的感觉真的很爽,加油吧兄弟们,这几天我凉了,从一天增加100粉丝到只有20几个了,┭┮﹏┭┮

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2144811.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【与C++的邂逅】--- C++的IO流

Welcome to 9ilks Code World (๑•́ ₃ •̀๑) 个人主页: 9ilk (๑•́ ₃ •̀๑) 文章专栏&#xff1a; 与C的邂逅 本篇博客我们来了解C中io流的相关知识。 &#x1f3e0; C语言输入输出 C语言中我们用到的最频繁的输入输出方式就是scanf ()与printf()。 sc…

数据处理与统计分析篇-day03-Numpy环境搭建

概述 python优势 Python作为当下最为流行的编程语言之一 可以独立完成数据分析的各种任务 数据分析领域里有海量开源库 机器学习/深度学习领域最热门的编程语言 在爬虫&#xff0c;Web开发等领域均有应用 常用开源库 numpy NumPy(NumericalPython) 是 Python 语言的一…

创客中国AIGC专题赛冠军天鹜科技:AI蛋白质设计引领者

“落霞与孤鹜齐飞,秋水共长天一色——这句出自《滕王阁序》的诗句,是我作为江西人熟记于心的佳句。它描绘的天地壮丽景色常浮现于我的脑海,正是这种豁达与壮观,启发我们将公司命名为‘天鹜科技’,我们希望将源自自然的蛋白质与现代科技的创新精神相结合,打造蛋白质设计与应用的…

OpenBayes 教程上新 | AI 时代的「神笔马良」,Hyper-SD 一键启动教程上线!

每次脑海中的画面栩栩如生&#xff0c;想画下来却难以下笔&#xff1f; 每次画完自己觉得非常像&#xff0c;但是旁人却一头雾水&#xff1f; 每次想用文生图&#xff0c;但不知道如何精确地输入 prompt&#xff1f; AI 时代的「神笔马良」Hyper-SD 来了&#xff01; 仅需简…

基本仪表放大器+基本电容耦合隔离放大器+OTA(基本OTA电路+OTA增益)

2024-9-18&#xff0c;星期三&#xff0c;21:37&#xff0c;天气&#xff1a;多云&#xff0c;心情&#xff1a;晴。大家中秋节都过的怎么样啊&#xff0c;如果没过爽也没有关系&#xff0c;因为再上八天班就能迎来10.1长假啦&#xff01;&#xff01;&#xff01;&#xff01;…

【机器学习】--- 自然语言推理(NLI)

引言 随着自然语言处理&#xff08;NLP&#xff09;的迅速发展&#xff0c;**自然语言推理&#xff08;Natural Language Inference, NLI&#xff09;**已成为一项重要的研究任务。它的目标是判断两个文本片段之间的逻辑关系。这一任务广泛应用于机器阅读理解、问答系统、对话…

五星级可视化页面(30):本系列最后一期,压轴出场。

不知不觉分享了30期高品质的五星级可视化大屏界面&#xff0c;该系列文章也该收尾了&#xff0c;本期为大家分享最后一批界面&#xff0c;我们下一个系列专辑见。

力扣之181.超过经理收入的员工

文章目录 1. 181.超过经理收入的员工1.1 题干1.2 准备数据1.3 题解1.4 结果截图 1. 181.超过经理收入的员工 1.1 题干 表&#xff1a;Employee -------------------- | Column Name | Type | -------------------- | id | int | | name | varchar | | salary | int | | mana…

W25QXX系列Flash存储器模块驱动代码

目录 W25QXX简介 硬件电路 W25Q128框图 Flash操作注意事项 驱动代码 W25QXX.h W25QXX.c W25QXX简介 W25Qxx系列是一种低成本、小型化、使用简单的非易失性存储器&#xff0c;常应用于数据存储、字库存储、固件程序存储等场景 存储介质&#xff1a;Nor Flash&#xff0…

Apache SeaTunnel Zeta引擎源码解析(三) Server端接收任务的执行流程

作者&#xff1a;刘乃杰 编辑整理&#xff1a;曾辉 引入 本系列文章是基于 Apache SeaTunnel 2.3.6版本&#xff0c;围绕Zeta引擎给大家介绍其任务是如何从提交到运行的全流程&#xff0c;希望通过这篇文档&#xff0c;对刚刚上手SeaTunnel的朋友提供一些帮助。 我们整体的文…

ios xib 子控件约束置灰不能添加约束

添加约束时发现置灰不可点的问题 layout切换为inferred&#xff0c;就可以添加约束了

[SIGGRAPH-24] CharacterGen

[pdf | code | proj] LRM能否用于3D数字人重建&#xff1f;问题在于&#xff1a;1&#xff09;缺少3D数字人数据&#xff1b;2&#xff09;重建任意姿态的3D数字人不利于后续绑定和驱动。构建3D数字人数据集&#xff1a;在VRoidHub上采集数据&#xff0c;得到13746个风格化角色…

青柠视频云——记一次大华摄像头公网语音对讲失败的问题分析

今天有客户反馈&#xff0c;使用大华摄像头接入青柠视频云&#xff0c;在公网环境下无法进行语音对讲&#xff0c;用户的设备是支持语音对讲的。 这是用户提供的注册截图&#xff0c;看起来也没什么问题&#xff0c;而且用户摄像头带有拾音功能和外放喇叭。 于是我们联系客户开…

Maple常用命令

1. 重启内核&#xff1a; restart 2. 化简式子 simplify(式子) 3. 引用前面出现的公式&#xff1a; CtrlL&#xff0c;在弹出的以下对话框中输入要引用的公式编号 4.

GHOST重装后DEF盘丢失:深度解析与高效数据恢复方案

在数字信息爆炸的时代&#xff0c;数据安全与恢复成为了每个计算机用户必须面对的重要课题。GHOST作为系统备份与恢复领域的佼佼者&#xff0c;以其快速、便捷的特点赢得了广泛的用户基础。然而&#xff0c;在使用GHOST进行系统重装的过程中&#xff0c;不少用户遭遇了DEF盘&am…

Qt_多元素控件

目录 1、认识多元素控件 2、QListWidget 2.1 使用QListWidget 3、QTableWidget 3.1 使用QListWidget 4、QTreeWidget 4.1 使用QTreeWidget 5、QGroupBox 5.1 使用QGroupBox 6、QTabWidget 6.1 使用QTabWidget 结语 前言&#xff1a; 在Qt中&#xff0c;控件之间…

【Linux】常见指令(3)

1.head指令 head指令用于显示文件的前几行内容&#xff0c;默认head指令打印其相应文件的开头10行。 使用方法&#xff1a;head [选项] [文件名] 常见的选项有&#xff1a; -n&#xff1a;指定显示文件的前几行&#xff0c;例如显示前五行可以输入命令“head -n 5 output.txt”…

ChromaDB教程_2024最新版(上)

前言 在上一篇&#xff08;快捷入口&#xff09;文章中&#xff0c;博主提到了一个向量存储&#xff0c;其中用到了Chroma数据库。代码示例如下&#xff1a; vectordb Chroma.from_documents(documentsdocs,embeddingembedding,persist_directoryvector_dir )这是基于langc…

智能车镜头组入门(一)车模的选择

这篇文章&#xff0c;我会简单的介绍下车模的、轮胎和负压的选择 今年的镜头组是自制车模&#xff0c;这比较考验学校之前参赛的经验。我们选择了某飞的mini车模。提供智能车方案的无非就两家&#xff0c;某飞和某邱&#xff0c;我们学校之前都用的是某飞的&#xff0c;在某飞…

功能测试干了2年,快要废了...

8年前刚进入到IT行业&#xff0c;到现在学习软件测试的人越来越多&#xff0c;所以在这我想结合自己的一些看法给大家提一些建议。 最近聊到软件测试的行业内卷&#xff0c;越来越多的转行和大学生进入测试行业&#xff0c;导致软件测试已经饱和了&#xff0c;想要获得更好的待…