面试经典150题——最长连续序列

news2025/2/25 15:53:56

"The only limit to our realization of tomorrow will be our doubts of today." - Franklin D. Roosevelt

green plant beside white desk

1. 题目描述

image-20240226101211338

2.  题目分析与解析

2.1 思路一

题目要求我们使用时间复杂度为O(n)的解决方案,那么肯定就不能排序了。因为排序算法不可能达到O(n)的时间复杂度。

常见的排序算法包括:

  1. 冒泡排序(Bubble Sort):时间复杂度为O(n^2)。

  2. 选择排序(Selection Sort):时间复杂度为O(n^2)。

  3. 插入排序(Insertion Sort):时间复杂度为O(n^2)。

  4. 归并排序(Merge Sort):时间复杂度为O(n log n)。

  5. 快速排序(Quick Sort):平均时间复杂度为O(n log n),最坏情况下为O(n^2)。

  6. 堆排序(Heap Sort):时间复杂度为O(n log n)。

  7. 希尔排序(Shell Sort):时间复杂度取决于间隔序列的选择,一般为O(n log n)到O(n^2)。

  8. 计数排序(Counting Sort):时间复杂度为O(n + k),其中k是输入范围。

  9. 桶排序(Bucket Sort):时间复杂度为O(n + k),其中k是桶的数量。

  10. 基数排序(Radix Sort):时间复杂度为O(d * (n + k)),其中d是数字的位数,k是基数。

那就只有考虑怎么用空间换时间,也就是把每一个走过的信息存储起来,方便后续使用,才有可能达到O(n)的时间复杂度。

现在我们再来审一下题目,既然题目要求连续,这是一个核心特征,我们就要根据这个特征作为抓手,看它有什么性质,从而给出可能的解决方案。

什么是连续?

题目中提到的连续指的是如 {1,2,3,4,5,6,7}这样的数字,它有什么特点?是不是显而易见,每个数字之差相差1。根据这个性质能否想出解决方案?

是可以的。假设我们遍历数组的每一个元素,我要看当前数字是否能和其它之前遍历过的数字是否连续,那么就只需要判断是否在之前遍历的集合中是否有某个数字与它的绝对值之差等于1,也就是:

  • 要么之前走过的某个数等于当前数+1

  • 要么之前走过的某个数等于当前数-1

那么就可以知道这两个数字是连续的,就可以把现有连续的数字长度加一,那么就说明我们对于走过的数字不仅仅需要存储数字本身,还需要存储当前序列的长度。而又因为每加入一个数字,都是在当前数字串的基础上加在前面或者加在后面,所以我们在存储序列长度时只需要将序列的头部与尾部更新长度即可:

对于上图所示,如果对于一个新加入的数字5,那么我们发现存在6=5+1,所以我们就把5拼接在6前面,此时需要更新该 5,6,7串的首尾两个数字的 序列长度,将其更新为3。但是注意,对于如下的情况:

image-20240227102613017

如上图所示,如果遍历到数字5,发现 5=4+1,而且 5+1=6,此时注意如果把该数字加入序列,那么就需要更新整体序列的第一个元素 3和最后一个元素 7的序列长度,将其变为5,如下:

image-20240227102914348

数字5处的对应的数字为1无需关系,只是一个默认数字。

所以根据以上的思路,我们就可以写出如下:

代码思路:

  1. 初始化:定义一个hashMap,用来存储走过的数字

  2. 遍历整个nums数组,判定当前数与某个走过的数字绝对值之差是否等于1

    image-20240227103126734

    就需要更新 3和7处也就是序列合并后的头与尾的长度

    那么就只需要更新当前数字和当前序列尾部数字对应的序列长度值

    • 最后判断如果存在走过的数字中某个数小于当前数1,那么就对应下述情况:

      此时只需要更新当前数字和当前序列头部数字对应的序列长度值

    • 然后判断如果存在走过的数字中某个数大于当前数1,那么就对应下述情况:

    • 先判断如果存在走过的数字中某个数大于当前数1,而且村子啊某个数小于当前数1,那么就对应下述情况:

  3. 遍历结束后hashMap中最大value即为结果

2.2 思路二

引自力扣官方

image-20240227124422206

实现过程动图如下:

recording

增加了判断跳过的逻辑之后,时间复杂度是多少呢?外层循环需要 O(n)的时间复杂度,只有当一个数是连续序列的第一个数的情况下才会进入内层循环,然后在内层循环中匹配连续序列中的数,因此数组中的每个数只会进入内层循环一次。根据上述分析可知,总时间复杂度为 O(n),符合题目要求。

在这里我主要给官方解析代码加一下注释,见后续代码实现。

3. 代码实现

image-20240227113446822

image-20240227113456394

思路一优化

其实我们记住问题的本质,要寻找的是最大序列的长度,对应上述解题思路就是最大的value,那我们为何不用一个变量来存储最大的value,这样就没必要在遍历结束后从hashMap中找最大value了。因此我们可以进行优化如下:

image-20240227123350130

image-20240227114059815

3.2 思路二

image-20240227125332924

image-20240227125315223

4. 相关复杂度分析

方法一:优化版(longestConsecutive_optimized

逻辑

  • 使用一个HashMap来追踪每个数字是否被访问过,以及它所在序列的长度。

  • 遍历给定数组,对于每个元素,检查其前后元素是否存在于HashMap中,并据此更新序列的长度。

时间复杂度

  • 遍历给定数组一次,时间复杂度为O(n)

  • 对于每个元素,查找和更新HashMap的操作均可在常数时间内完成(假设HashMap的操作为O(1))。

  • 因此,总的时间复杂度是O(n)

空间复杂度

  • 使用了一个HashMap来存储至多n个元素及其对应的序列长度,因此空间复杂度为O(n)

方法二:基于HashSet(longestConsecutive2

逻辑

  • 首先,将所有元素存入HashSet中,以便快速查找任意元素是否存在。

  • 然后,遍历HashSet,对于每个元素,如果它的前一个元素不存在于HashSet中,则认为它是序列的起点,并从这个点开始尝试构建连续序列。

时间复杂度

  • 将所有元素添加到HashSet中的时间复杂度为O(n)

  • 遍历HashSet的时间复杂度也为O(n)。对于每个序列的起始元素,我们可能会在HashSet中进行多次查找(最坏情况下为序列长度次),但每个元素最多被访问两次(一次作为某个序列的一部分,另一次作为尝试找下一个元素)。因此,整体的时间复杂度依然是O(n)

空间复杂度

  • 使用了一个HashSet来存储所有元素,空间复杂度为O(n)

总结

两种方法都具有O(n)的时间复杂度和O(n)的空间复杂度。主要区别在于实现细节和具体操作。优化版方法直接在遍历过程中更新和记录序列的长度,而基于HashSet的方法则是先构建一个集合,再通过这个集合找到所有可能的序列起点,从而构建序列。

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

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

相关文章

【Web】CTFSHOW XXE刷题记录(全)

目录 web373 web374 web375 web376 web377 web378 前置知识先看这篇文章&#xff1a;XXE漏洞学习 用的多的就是外部实体声明和参数实体声明 web373 有回显的xxe <!DOCTYPE test [ <!ENTITY xxe SYSTEM "file:///flag"> ]> <z3r4y> <ct…

Jupyterlab 和 JupyternoteBook 修改默认路径

Jupyterlab 和 JupyternoteBook 修改默认路径 在使用 JupyterLab 或 Jupyter Notebook 进行数据分析、机器学习项目时&#xff0c;经常会遇到需要修改默认工作目录的需求。默认情况下&#xff0c;JupyterLab 和 Jupyter Notebook 会在启动时打开你的用户目录&#xff08;例如&…

Stable Diffusion 绘画入门教程(webui)-ControlNet(Seg)

上篇文章介绍了深度Depth&#xff0c;这篇文章介绍下seg&#xff08;Segmentation&#xff09; 意思为语义分割, 通俗理解就是把图中的不同物体元素按类别不同&#xff0c;标为不同的颜色&#xff0c;不同的颜色代表不同的元素类别&#xff0c;如下图&#xff0c;左边为原图&a…

Springboot 多级缓存设计与实现

&#x1f3f7;️个人主页&#xff1a;牵着猫散步的鼠鼠 &#x1f3f7;️系列专栏&#xff1a;Java全栈-专栏 &#x1f3f7;️个人学习笔记&#xff0c;若有缺误&#xff0c;欢迎评论区指正 前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&…

夯实数据管理基础,激活数据资产价值—数据资产运营解决方案介绍

“数据是企业的核心战略资产”已然成为共识。然而金融行业数据资产运营目前普遍存在“锚不定”&#xff0c;缺少企业级数据战略&#xff0c;业数融合不足&#xff1b;“驱不动”&#xff0c;缺少业务和运营思维&#xff0c;以技术为驱动的推进模式&#xff0c;缺乏升级活力&…

web安全学习笔记【17】——信息打点(7)

信息打点-APP资产&知识产权&应用监控&静态提取&动态抓包&动态调试 #知识点&#xff1a; 1、业务资产-应用类型分类 2、Web单域名获取-接口查询 3、Web子域名获取-解析枚举 4、Web架构资产-平台指纹识别 ------------------------------------ 1、开源-CMS指…

电子病历系统

电子病历系统 获取源码——》公主号&#xff1a;计算机专业毕设大全

面试官: 反射了解么?

目录 反射 什么是反射 ? 获取Class对象的四种方式 反射相关API 类对象常用API Filed常用API Method常用API Constructors常用API 反射的使用场景? 反射的实现原理 ?(todo) 反射为什么这么慢 ? 反射的优缺点 反射中&#xff0c;Class.forName和ClassLoader的区别…

基于JAVA的就医保险管理系统 开源项目

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 科室档案模块2.2 医生档案模块2.3 预约挂号模块2.4 我的挂号模块 三、系统展示四、核心代码4.1 用户查询全部医生4.2 新增医生4.3 查询科室4.4 新增号源4.5 预约号源 五、免责说明 一、摘要 1.1 项目介绍 基于JAVAVue…

LZO无损数据解压缩IP,高性能版本

LZOAccel-D LZO Data Decompression Core/无损数据解压缩IP Core LZOAccel-D是一个无损数据解压缩引擎的FPGA硬件实现&#xff0c;兼容LZO 2.10标准。 Core接收压缩的输入数据块&#xff0c;产生解压缩后的数据块。Core分析数据块的头和尾&#xff0c;检查输入数据块的错误&…

第十一届上海国际蒸发结晶展将于12月11-13日上海举办

展会介绍&#xff1a; 蒸发结晶展是我司2013年创办。通过这几年我司不断努力&#xff0c;得到行业逐步认可&#xff0c;受到行业和企业的一致肯定和积极评价&#xff0c;具有较高的知名度和行业影响力&#xff0c;也具有一定国际知名度的蒸发结晶专业会展活动之一。蒸发结晶技…

网络安全与IP安全网络安全

网络安全与IP安全网络安全 网络安全 是指网络系统的硬件&#xff0c;软件以及系统中的数据收到的保护。 保护的基本属性为&#xff1a;机密性&#xff0c;身份认证&#xff0c;完整性和可用性&#xff1b; 基本特征&#xff1a;相对性&#xff0c;时效性&#xff0c;相关性…

springboot自写插件封包

在Spring Boot中自写插件或封包&#xff08;通常指的是创建自定义的starter&#xff09;是一种常见的做法&#xff0c;用于将一系列相关的配置和组件打包成一个独立的模块&#xff0c;从而简化依赖管理和配置过程。以下是一个简单的步骤&#xff0c;指导你如何创建一个自定义的…

2024牛客寒假算法基础集训营2

目录 A.Tokitsukaze and Bracelet B.Tokitsukaze and Cats C.Tokitsukaze and Min-Max XOR D.Tokitsukaze and Slash Draw E and F.Tokitsukaze and Eliminate (easy)(hard) G.Tokitsukaze and Power Battle (easy) 暂无 I.Tokitsukaze and Short Path (plus) J.Tokits…

P0故障应对策略之:为什么P0故障难以排查

与大模型探讨P0故障 P0级故障&#xff0c;作为系统中最严重的故障&#xff0c;它们的发生往往带来灾难性的后果和巨大的损失。同时&#xff0c;这类故障的排查与修复也往往复杂而棘手&#xff0c;对整个团队的经验、综合能力、应急处置流程都是巨大的挑战。 排查P0级故障的过程…

react useMemo 用法

1&#xff0c;useCallback 的功能完全可以由 useMemo 所取代&#xff0c;如果你想通过使用 useMemo 返回一个记忆函数也是完全可以的。 usecallback(fn,inputs)is equivalent to useMemo(()> fn, inputs). 区别是:useCallback不会执行第一个参数函数&#xff0c;而是将它返…

linux查看socket信息

netstat netstat 是一个用于显示网络相关信息的命令行工具。它可以显示当前系统的网络连接状态、路由表、接口统计信息等。 下面是一些常见的 netstat 命令选项和用法&#xff1a; 显示所有活动的网络连接&#xff1a; netstat -a 显示所有正在监听的端口&#xff1a; ne…

TiDB之分布式数据库TiDB 操作管理规范【附可下载文档】

一、 目的 为了在软件生命周期内规范数据库相关的设计、开发、运维工作,便于不同团队之间的沟通及协调,制定此文档,以期在相关规范上达成共识和默契,提升相关环节的工作效率及系统的可维护性。同时好的规范,在执行的时候可以培养出好的习惯,好的习惯是软件质量的很好保证…

飞行棋网站源码

最近抖音很火的情侣飞行棋网站源码&#xff0c;这款情侣飞行棋提供了丰富的游戏玩法&#xff0c;可以为情侣、朋友或家人带来欢乐的游戏体验。无论是在家中&#xff0c;还是在聚会、旅行等场合&#xff0c;都可以轻松启动该网站&#xff0c;共同享受游戏的乐趣。 源码获取搜一搜…

【BUG 记录】史诗级 BUG - MYSQL 删库删表却没有备份如何恢复数据

【BUG 记录】史诗级 BUG - MYSQL 删库删表却没有备份如何恢复数据 1. 问题描述2. 解决方案&#xff08;binlog&#xff09;2.1 构造测试环境2.2 查看 MySQL 环境是否开启 binlog2.3 查看所有的 binlog 日志记录2.4 查看当前正在使用的是哪一个 binlog 文件2.5 查看此时的 binlo…