news 2026/6/23 8:47:32

【LeetCode Hot100】128. 最长连续序列 - Java O(n) 解法详解

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
【LeetCode Hot100】128. 最长连续序列 - Java O(n) 解法详解

📝 前言

今天在刷 LeetCode 热题 100 时,碰到了第 128 题“最长连续序列”。这是一道非常经典的题目,考察的重点是如何在不排序的情况下,利用哈希表在 O(n) 的时间复杂度内完成查找。

乍一看这道题如果用Arrays.sort()排序后遍历,时间复杂度是 O(nlog n),但这道题明确要求O(n),所以必须换一种思路。记录一下我的解题心得和最终代码。

📚 题目描述

给定一个未排序的整数数组nums,找出数字连续的最长序列(不要求序列元素在原数组中连续)的长度。

示例 1:

输入: nums = [100,4,200,1,3,2]

输出: 4

解释: 最长数字连续序列是 [1, 2, 3, 4]。它的长度为 4。

提示:

  • 请设计并实现时间复杂度为 O(n) 的算法。


💡 解题思路

1. 为什么不能排序?

题目硬性要求时间复杂度为 O(n)。我们知道标准的排序算法(如快排、归并)最快也是 O(nlog n),所以排序这条路走不通。我们需要一种能够快速查找的数据结构,哈希表 (HashSet)是最佳选择。

2. 核心逻辑:去重与定位“起点”

我们可以分两步走:

  1. 去重与存储:先把所有数字放入HashSet中,这样我们不仅去除了重复元素,还能在 O(1) 的时间内判断一个数是否存在。

  2. 寻找序列起点

    • 如果我们对集合中的每一个数x都去尝试向后枚举 (x+1,x+2...),时间复杂度最坏会达到 O(n^2)。

    • 关键优化点:我们只从序列的起点开始查找。

    • 如何判断起点?如果一个数x,它的前驱x-1在集合中,那么x一定是某个连续序列的第一个数。

    • 只有当x是起点时,我们才开始向后匹配x+1,x+2等等,统计长度。

3. 一个小优化

在统计过程中,如果当前找到的最长序列长度已经超过了哈希表中剩余元素的一半(或者总数的一半),其实就可以提前结束循环了,因为剩下的元素数量不可能凑出更长的序列。


💻 代码实现 (Java)

代码相比官方题解做了变量名的语义化修改,使其更符合工程规范,方便阅读。

class Solution { public int longestConsecutive(int[] nums) { // 1. 预处理:将数组元素放入 HashSet,实现去重和 O(1) 查询 Set<Integer> numSet = new HashSet<>(); for (int num : nums) { numSet.add(num); } int maxLen = 0; int totalNums = numSet.size(); // 2. 遍历集合中的每个元素 for (int num : numSet) { // 核心剪枝逻辑: // 只有当 num-1 不存在时,num 才是一个连续序列的【起点】 // 如果 num-1 存在,说明 num 已经被计算过了,直接跳过 if (!numSet.contains(num - 1)) { int currentNum = num; int currentLen = 1; // 从起点开始,不断向后寻找连续的数字 while (numSet.contains(currentNum + 1)) { currentNum += 1; currentLen += 1; } // 更新最大长度 maxLen = Math.max(maxLen, currentLen); // 【可选优化】:如果当前找到的长度已经超过总数的一半, // 那么剩下的元素不可能组成更长的序列,直接退出 if (maxLen > totalNums / 2) { break; } } } return maxLen; } }

🔍 复杂度分析

  • 时间复杂度:O(n)

    • 虽然代码里有一个while循环嵌套在for循环里,但仔细分析会发现:由于if (!numSet.contains(num - 1))的限制,数组中的每个数最多只会被访问两次(一次是作为序列起点被访问,一次是作为序列的一部分被内部while访问)。

    • 因此总的操作次数是线性的。

  • 空间复杂度:O(n)

    • 我们需要一个HashSet来存储数组中的元素,以空间换时间。

🎯 总结

这道题是哈希表运用的典范。解决很多 O(n) 复杂度问题的秘诀往往就在于“如何避免重复计算”。在这道题里,通过判断x-1是否存在,精准地锁定了每个序列的头部,从而避免了大量的无效枚举。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/23 6:58:20

Turnitin系统查英文AI率多少为正常?报告显示星号*%怎么办?

很多学校和杂志社都在说需要检测论文AI率&#xff0c;但是论文AI率多少才算正常呢&#xff1f; Turnitin系统AI检测系统&#xff1a;https://students-turnai.similarity-check.com/ 今天这篇内容就给大家分享一下。 很多同学使用Turnitin系统检测了英文论文AI率之后&#x…

作者头像 李华
网站建设 2026/6/22 21:36:24

暖通净化空调恒温恒湿项目:PLC 与触摸屏上位机程序探秘

暖通净化空调恒温恒湿项目包括PLC程序和触摸屏上位机程序。 标准化很好的内部用的函数都封装成了标准块一套很好的学习资料。在暖通净化空调恒温恒湿项目里&#xff0c;PLC 程序和触摸屏上位机程序就像项目运转的左膀右臂&#xff0c;承担着关键任务。先聊聊 PLC 程序&#xff…

作者头像 李华
网站建设 2026/6/22 22:05:55

第30章 Shell 正则表达式实战:精准匹配字符串、日志与配置项

本章导语:正则表达式是文本处理的"瑞士军刀",是 Linux 系统管理和数据处理的核心技能。掌握正则表达式,你将能够精准匹配和处理各种复杂的文本模式,从日志分析到配置文件管理,从数据清洗到格式验证,无所不能。本章将通过丰富的实战案例,帮助你彻底掌握正则表达…

作者头像 李华
网站建设 2026/6/22 3:00:23

音视频学习(七十二):视频压缩:分块与预处理

分块与预处理是视频压缩&#xff08;编码&#xff09;流程的起点&#xff0c;它的目标是将原始的、高冗余的视频数据转换成适合高效压缩的格式和基本处理单元。这一阶段的工作质量直接影响后续运动估计、变换编码和量化等步骤的效率和最终的压缩比与图像质量。 预处理的核心目标…

作者头像 李华
网站建设 2026/6/21 21:18:31

AMD Ryzen性能调优:快速掌握处理器调试工具的使用技巧

AMD Ryzen性能调优&#xff1a;快速掌握处理器调试工具的使用技巧 【免费下载链接】SMUDebugTool A dedicated tool to help write/read various parameters of Ryzen-based systems, such as manual overclock, SMU, PCI, CPUID, MSR and Power Table. 项目地址: https://gi…

作者头像 李华