41.寻找缺失的第一个正数:原地哈希算法详解

文章目录

    • 引言
    • 问题描述
    • 方法思路:原地哈希算法
      • 算法步骤
    • 完整代码实现
    • 关键代码解析
    • 复杂度分析
    • 示例说明
    • 总结

引言

在算法面试和数据处理中,寻找缺失的第一个正数是一个经典问题。题目要求给定一个未排序的整数数组,找到其中缺失的最小正整数,且时间复杂度需为O(n),空间复杂度为O(1)。本文将详细解析如何通过原地哈希算法高效解决这一问题,并提供完整的Java代码实现。


问题描述

给定一个包含n个整数的数组nums,找出其中没有出现的最小的正整数
示例

  • 输入:nums = [1,2,0],输出:3
  • 输入:nums = [3,4,-1,1],输出:2

限制条件

  • 时间复杂度:O(n)
  • 空间复杂度:O(1)

方法思路:原地哈希算法

传统方法(如排序或哈希表)无法满足空间复杂度要求。原地哈希的核心思想是将数值映射到数组索引上,通过交换操作将每个正整数放到其“正确位置”。

算法步骤

  1. 处理数组元素

    • 遍历数组,若当前元素nums[i]是正整数且在[1, n]范围内,则将其交换到索引nums[i]-1的位置。
    • 使用while循环确保交换后的新元素也被正确处理。
  2. 查找缺失值

    • 再次遍历数组,第一个满足nums[i] != i+1的位置即为缺失的最小正数。
    • 若所有位置均正确,则缺失值为n+1

完整代码实现

public class FirstMissingPositive {public int firstMissingPositive(int[] nums) {int n = nums.length;// 原地哈希:将数值x交换到索引x-1的位置for (int i = 0; i < n; i++) {while (nums[i] >= 1 && nums[i] <= n && nums[nums[i] - 1] != nums[i]) {swap(nums, i, nums[i] - 1);}}// 查找第一个不满足nums[i] == i+1的位置for (int i = 0; i < n; i++) {if (nums[i] != i + 1) {return i + 1;}}return n + 1;}// 交换数组中的两个元素private void swap(int[] nums, int i, int j) {int temp = nums[i];nums[i] = nums[j];nums[j] = temp;}
}

关键代码解析

  1. 原地哈希处理

    while (nums[i] >= 1 && nums[i] <= n && nums[nums[i] - 1] != nums[i]) {swap(nums, i, nums[i] - 1);
    }
    
    • 条件1nums[i]必须是有效正整数(1 ≤ x ≤ n)。
    • 条件2:若目标位置的值已正确,则无需交换。
    • 循环处理:交换后,新的nums[i]可能仍需调整,因此使用while而非if
  2. 查找缺失值

    if (nums[i] != i + 1) {return i + 1;
    }
    
    • 第一个不满足nums[i] == i+1的索引i,其对应的i+1即为缺失值。

复杂度分析

  • 时间复杂度:O(n)
    每个元素最多被交换一次,两次遍历时间复杂度均为O(n)。
  • 空间复杂度:O(1)
    仅使用常数级别的额外空间。

示例说明

以输入nums = [3,4,-1,1]为例:

  1. 原地哈希处理
    • 遍历到3,将其交换到索引2的位置,数组变为[-1,4,3,1]
    • 遍历到4,将其交换到索引3的位置,数组变为[-1,1,3,4]
    • 遍历到1,将其交换到索引0的位置,数组变为[1,-1,3,4]
  2. 查找缺失值
    • 索引1的值为-1,不满足nums[1] == 2,返回2

总结

原地哈希算法通过索引映射元素交换,将时间复杂度优化至O(n),空间复杂度优化至O(1)。该方法不仅高效,还能帮助深入理解数组索引与数值之间的关系。掌握这一算法,可轻松应对类似的高频面试题。

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

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

相关文章

matlab 中function的用法

matlab 中function的用法 前言介绍1. 基本语法示例&#xff08;1&#xff09;可以直接输出&#xff08;2&#xff09;调用函数 2.输入参数和输出参数示例多输入参数和输出参数定义一个函数&#xff0c;计算两个数的和与差&#xff1a;调用该函数&#xff1a; 3. 默认参数示例 4…

HarmonyOS开发之基于子窗口实现应用内悬浮窗

鸿蒙开发&#xff1a;基于子窗口实现应用内悬浮窗(含完整代码示例) 在现代移动应用中&#xff0c;悬浮窗/悬浮球是一种非常实用的交互方式&#xff0c;常用于展示快捷入口、实时通知、视频播放等场景。例如&#xff1a; 聊天应用中的小助手按钮视频应用的画中画功能游戏或工具类…

可以下载blender/fbx格式模型网站

glbxz.com glbxz.com可以下载blender/fbx格式模型。当然里面有免费的

250505_HTML

HTML 1. HTML5语法与基础标签1.1 HTML5特性1.1.1 空白折叠现象1.1.2 转义字符 1.2 HTML注释1.3 基础标签1.3.1 div标签1.3.2 标题标签1.3.3 段落标签1.3.4 title1.3.5 meta 1.4 html骨架1.4.1 DTD1.4.2 html标签1.4.3 head与body标签 1.5 div标签详解1.5.1 常见class类名 1.6 列…

数据封装的过程

数据的封装过程 传输层 UDP 直接将数据封装为UDP数据报​&#xff0c;添加UDP头部&#xff08;8B&#xff09;。 要点&#xff1a; UDP首部简单&#xff0c;无连接不可靠、无重传、无拥塞控制&#xff0c;适用于实时性要求较高的通讯&#xff1b;不需要源端口或不想计算检…

面向AGI的语言认知操作系统形式化模型

邹晓辉融智学语言数据库体系的数学表达 ——面向AGI的语言认知操作系统形式化模型 1. 基础定义与符号系统 设语言宇宙 L 为所有语言要素的集合&#xff0c;其结构可分解为&#xff1a; LY(言)U(语)A(用) 其中&#xff1a; YPGS &#xff08;音/形/义三元组&#xff09; U⋃…

基于 Spring Boot 瑞吉外卖系统开发(十)

基于 Spring Boot 瑞吉外卖系统开发&#xff08;十&#xff09; 修改菜品 修改菜品是在原有的菜品信息的上对菜品信息进行更新&#xff0c;对此修改菜品信息之前需要将原有的菜品信息在修改界面进行展示&#xff0c;然后再对菜品信息进行修改。 修改菜品分为回显菜品信息和更…

Three.js和WebGL区别、应用建议

Three.js 和 WebGL 是用于在浏览器中创建 3D 图形的两种技术,它们之间有明显的区别和适用场景。 对于一般数据展示和模型展示而言,应用更多的是three.js,毕竟相对学习成本来说webGL跟高,需要投入更多的精力和基础功能的开发和验证上。而three.js封装了webGL的功能,开发相对…

【Vue】移动端开发(Uni-app、Taro)

个人主页&#xff1a;Guiat 归属专栏&#xff1a;Vue 文章目录 1. Uni-app 与 Taro 简介1.1 什么是 Uni-app&#xff1f;1.2 什么是 Taro&#xff1f;1.3 Uni-app vs Taro&#xff08;对比图&#xff09; 2. 项目初始化与目录结构2.1 初始化 Uni-app 项目2.2 初始化 Taro 项目&…

自定义SpringBoot Starter-笔记

SpringBoot Starter的介绍参考&#xff1a; Spring Boot Starter简介-笔记-CSDN博客。这里介绍如何自定义一个springBoot Starter。 1. 项目结构 创建一个 Maven 项目&#xff0c;结构如下&#xff1a; custom-spring-boot-starter-demo/ ├── custom-hello-jdk/ # jdk模…

linux >!

Linux 中 >! 符号的含义与用法 ‌基本定义‌在 Linux Shell 中,>! 是由 > 和 ! 组合的特殊符号,主要用于 ‌强制覆盖文件‌。其行为与常规的 > 类似,但额外添加了忽略潜在限制的功能。 ‌典型场景‌绕过 noclobber 限制‌: 若 Shell 启用了 noclobber 选项(默…

共铸价值:RWA 联合曲线价值模型,撬动现实资产生态

摘要 本文提出了一种针对真实资产&#xff08;RWA&#xff09;产业的联合曲线激励模型&#xff0c;将劳动与数据贡献映射为曲线价值&#xff0c;并基于固定档位与指数衰减奖励发放总计 2.1亿积分。该模型结合了去中心化定价与平滑递减机制&#xff0c;不仅为早期贡献者提供更高…

java安全入门

文章目录 java基础知识this变量方法可变参数构造方法继承的关键字protected super阻止继承方法重载向上转型和向下转型多态抽象接口static静态字段default方法 包final内部类 java序列化与反序列化反射urldns链动态代理类加载器&#xff08;ClassLoader&#xff09;双亲委派模型…

前端基础之《Vue(14)—组件通信(1)》

一、什么是组件通信 1、通信是组件或模块之间的数据交互。 2、组件多重通信就形成了数据流&#xff0c;数据流管理的优劣决定了产品能否上线&#xff0c;返工是否频繁的问题。 3、Vue中有哪些常见的通信方案&#xff1f; 组件树的概念&#xff1a; 在Vue中&#xff0c;组件…

el-row el-col

参考layout布局 Element - The worlds most popular Vue UI frameworkElement&#xff0c;一套为开发者、设计师和产品经理准备的基于 Vue 2.0 的桌面端组件库https://element.eleme.cn/#/zh-CN/component/layout#row-attributes 一行可以看做24个 Element UI 中的 el-row 是…

Socket-TCP

在TCP/ip协议中&#xff0c;用源IP、源端口号、目的IP、目的端口号、协议号这样一个五元组来标识一个通信&#xff01; 端口号范围划分 0 - 1023: 知名端口号&#xff0c;HTTP&#xff0c;FTP&#xff0c;SSH 等这些广为使用的应用层协议&#xff0c;他们的端口号都是固定的。…

大数据Spark(五十八):Spark Pi介绍

文章目录 Spark Pi介绍 Spark Pi介绍 Spark Pi是Apache Spark官方提供的一个示例程序&#xff0c;该案例使用 Spark 进行分布式计算&#xff0c;通过蒙特卡罗方法估算圆周率&#xff08;π&#xff09;的值&#xff0c;其估算π原理如下&#xff1a; 上图中&#xff0c;正方形…

Doris索引机制全解析,如何用高效索引加速数据分析

在当今大数据时代&#xff0c;企业对于实时数据分析的需求呈现爆发式增长。面对动辄PB级的数据量和秒级响应的业务诉求&#xff0c;传统数据库系统往往力不从心。Apache Doris作为新一代MPP分析型数据库&#xff0c;凭借其独特的索引机制&#xff0c;在京东、美团等企业的实时数…

基于SpringBoot + Vue 的作业管理系统

产品包含&#xff1a; 项目源码数据库文件论文ppt 技术栈 架构: B/S、MVC 系统环境&#xff1a;Windows/Mac 开发环境&#xff1a;IDEA、JDK1.8、Maven、Mysql 技术栈&#xff1a;Java、Mysql、SpringBoot、Mybatis、Vue 功能模块 用户模块&#xff1a;学生用户、管理员、…

HCL(HashiCorp Configuration Language)是一种结构化配置语言

HCL&#xff08;HashiCorp Configuration Language&#xff09;是一种结构化配置语言&#xff0c;语法简洁且可读性强&#xff0c;广泛用于 Docker Buildx Bake、Terraform、Nomad 等工具的配置。以下是其核心语法规则和示例&#xff1a; 1. 基础结构 HCL 使用 块&#xff08;…