【忍者算法】从图书馆编目到数组搜索:探索缺失的第一个正整数|LeetCode 41 缺失的第一个正整数

news/2025/2/6 23:39:25/文章来源:https://www.cnblogs.com/renzhesuanfa/p/18701895

从图书馆编目到数组搜索:探索缺失的第一个正整数

生活中的算法

想象你是一位图书馆管理员,正在整理一排连续编号的图书。这些书应该从1号开始按顺序排列,但是有些编号的书不见了。你的任务是找出第一个缺失的编号。这就像是在做点名,发现第一个没来上课的同学。

这个场景在生活中很常见。比如:

  • 餐厅服务员查看哪个桌号是第一个空位
  • 停车场管理员寻找第一个空闲的车位号
  • 学校给新生分配第一个未使用的学号
  • 医院为病人安排第一个可用的就诊序号

问题描述

LeetCode第41题"缺失的第一个正整数"是这样描述的:给你一个未排序的整数数组 nums,请你找出其中没有出现的最小的正整数。请你实现时间复杂度为 O(n) 并且只使用常数级别额外空间的解决方案。

例如:

输入:nums = [3,4,-1,1]
输出:2
解释:数组中有1,3,4,所以第一个缺失的正整数是2。输入:nums = [7,8,9,11,12]
输出:1
解释:数组中没有1,所以缺失的第一个正整数是1。

最直观的解法:排序后遍历

就像整理图书时,先把所有的书按编号排好序,然后从1开始检查,看哪个编号最先空缺。

让我们用一个简单的例子来理解:

原数组:[3,1,4,-1]
1. 先排序(只考虑正数):[1,3,4]
2. 从1开始检查:- 1存在- 2不存在,找到答案!

优化解法:原地标记

仔细思考会发现一个关键点:如果数组长度为n,那么答案一定在[1, n+1]范围内。就像有10个学生的班级,第一个缺席的学号最大不会超过11。

我们可以把数组本身作为标记板,把每个数放到它应该在的位置上(就像把每本书放到对应的编号位置)。

原地标记的原理

  1. 把每个在[1,n]范围内的数x放到索引x-1的位置
  2. 再次遍历,第一个不在对应位置的数就指示了缺失的最小正整数
    这就像是:
  • 先把每本书放到它编号对应的书架位置
  • 然后从1号位置开始检查,找到第一个空位置

示例演示

用nums = [3,4,-1,1]来说明:

1. 开始移动元素:[3,4,-1,1] 把3放到索引2[3,4,3,1] 把4放到索引3[3,1,3,4] 把1放到索引0[1,3,3,4]2. 再次遍历,检查每个位置:位置0:期望1,实际1,正确位置1:期望2,实际3,找到答案2!

Java代码实现

public int firstMissingPositive(int[] nums) {int n = nums.length;// 第一遍遍历:把每个数放到它应该在的位置for (int i = 0; i < n; i++) {// 当前数在有效范围内,且不在正确位置上while (nums[i] > 0 && nums[i] <= n && nums[i] != nums[nums[i] - 1]) {// 交换到正确位置int temp = nums[nums[i] - 1];nums[nums[i] - 1] = nums[i];nums[i] = temp;}}// 第二遍遍历:找出第一个不在正确位置上的数for (int i = 0; i < n; i++) {if (nums[i] != i + 1) {return i + 1;}}// 如果都正确,返回n+1return n + 1;
}

解法比较

让我们比较这两种方法:

排序后遍历:

  • 时间复杂度:O(nlogn)
  • 空间复杂度:O(1)
  • 优点:思路直观,易于理解
  • 缺点:不满足时间复杂度要求

原地标记:

  • 时间复杂度:O(n)
  • 空间复杂度:O(1)
  • 优点:满足所有要求,不需要额外空间
  • 缺点:实现略微复杂,需要仔细处理边界情况

解题技巧总结

这道题给我们的启示:

  1. 思考数据的取值范围很重要
  2. 有时候可以把数组本身当作标记数组使用
  3. 位置和值的对应关系常常能带来灵感
  4. 不要害怕修改原数组,有时这是提高效率的关键

类似的问题还有:

  • 数组中重复的数字
  • 找到所有数组中消失的数字
  • 寻找重复数

小结

通过缺失的第一个正整数这道题,我们学会了一个重要的思维方式:有时候看似需要额外空间的问题,可以通过巧妙利用输入数组本身来解决。这种思维不仅在这道题中有用,在处理其他需要标记或计数的问题时也很有启发。记住,当遇到需要找寻特定范围内缺失数字的问题时,考虑能否利用数组本身来存储信息!


作者:忍者算法
公众号:忍者算法

完整题解项目 :https://github.com/ninjaAlgorithm/LeetCode-Solutions-Hot-100

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

L0G2-Python 前置基础

L0G2-Python 前置基础 1. 任务一 Leetcode 383 通过的截图通过的代码 class Solution(object):def canConstruct(self, ransomNote, magazine):""":type ransomNote: str:type magazine: str:rtype: bool"""# 只使用 defaultdict(int) 避免 if 语…

L0G3-Git 基础知识

L0G3-Git 基础知识 1. 任务一 破冰活动之自我介绍 1.1 配置 git 直接采用全局设置的方式 git config --global user.name "sakuralggm" git config --global user.email "myemail@qq.com"查看全局配置 git config --global --list1.2 fork 项目1.3 克隆项目…

L0G4-玩转HF魔搭魔乐社区

L0G4-玩转HF/魔搭/魔乐社区 1. Hugging Face平台 1.1 任务一 模型下载 1.1.1 创建github codespace 访问github codespace,并使用Jupyter Notebook模板。创建完成后,进入“Create frome a template”页面对已经创建的codespace选择“open in visual studio code”点击后会经过…

2024 年度总结:与自己对话

搬运自洛谷博客,原发表时间:2024-12-27(实际写成:12-31)\[\mathsf{那跑过去的昼夜\ \ 是孤独的修炼\qquad——\ 林俊杰《黑夜问白天》} \] 孤独而不孤独.又来到写年度总结的时刻了。这一年无疑比去年更具故事的戏剧性与心境的跌宕起伏了。 被各种事件聚在一起搅和,高一上…

breach1靶机渗透学习

靶机下载 https://www.vulnhub.com/ 找Breach-1.0,不过这个是老靶机了,可以看网上其他博客附带有靶机下载的网盘链接,靶机官网访问很慢而且网站好像出问题了排版很乱,不打也罢。 readme.txt: First in a multi-part series, Breach 1.0 is meant to be beginner to interm…

lxl数据结构讲义例题泛做

搬运自洛谷博客,原发表时间:2022-12-06 网课人在家摆烂太悲伤了,于是做一些数据结构题。 本博客和ds学习笔记的区别:后者收录小清新/有趣题。 目前拥有的讲义:感觉数据结构题评紫的巨大多,良莠不齐。 可能会有非常简单的题加入这里,勿喷。 完成度指数:0%表示一半以上核…

电路板地与外壳地之间接一个电阻和一个电容的作用.

原理 原理图如下:我们知道外壳基本都是金属属性,外壳会预留一个螺丝孔,预留的螺丝孔是为了跟电路板地连接在一起,电路设计如原理图所示,这样做有什么作用呢?作用如下:外壳地如果不稳定或者有静电之类的,如果与电路板地直接连接,就会打坏电路板芯片,加入电容,就能把低…

晶体管

PCB设计中晶体管的学习与应用 在电子电路设计中,晶体管作为一种重要的有源器件,扮演着放大、开关、稳压等多种角色。本文将带您了解晶体管的基础知识,以及如何在PCB设计中正确选择和应用晶体管。 晶体管基础 晶体管主要有两种类型:双极型晶体管(BJT)和场效应晶体管(FET)…

Tailwin CSS 基础下篇

Tailwin CSS 进阶上篇 方便自己查看,仅做摘录,非原创。 原文链接 听说你还不会 Tailwind CSS(基础上篇)Tailwind CSS 的基础使用:从宽度高度开始逐步展现 Tai - 掘金 (juejin.cn) 听说你还不会 Tailwind CSS(基础中篇)Tailwind CSS 的基础使用:包含伪类、伪元素、flex …

ctfshow密码学wp

打hgame打傻了来洗洗脑子( 1、密码学签到2、 jsfuck扔控制台3、 aaencode颜文字4、 ctfrsatoolsfrom Crypto.Util.number import * from gmpy2 import * p=447685307 q=2037 e=17 phi=(p-1)*(q-1) d=inverse(e,phi) print(d) 5、 交intimport gmpy2,libnum f…

异动拉升之量子图形矩阵

一.量子图形矩阵通过异动拉升算法检测出异动拉升的股票之后,需要图形化显示股票分时曲线,帮助观察股票异动之后的走势,量子图形矩阵的功能就是将异动拉升的股票,按照冠绝榜中的排序,从上到下依次展示在图形矩阵中,可以选择4,9,16个窗口,在窗口没有锁住的情况下,新出现…

MySQL主从复制 —— 作用、原理、数据一致性,异步复制、半同步复制、组复制

MySQL主从复制 作用、原理—主库线程、I/O线程、SQL线程;主从同步要求,主从延迟原因及解决方案;数据一致性,异步复制、半同步复制、组复制文章目录 一、作用 二、原理 三、同步数据一致性3.1 主从同步要求3.2 主从延迟原因、直接表现3.3 减少主从延迟的方案3.4 数据一致性…