当前位置: 首页 > ai >正文

LeetCode经典题解:128、最长连续序列

“最长连续序列”是一道极具代表性的数组处理问题, 本文将带你从直观思路出发,逐步推导出最优解法,并通过场景化记忆技巧掌握核心逻辑。

一、题目描述

题目:给定一个未排序的整数数组 nums,找出数字连续的最长序列(不要求序列元素在原数组中连续)的长度。
要求:时间复杂度为 O(n)。

示例
输入:nums = [100, 4, 200, 1, 3, 2]
输出:4
解释:最长连续序列是 [1, 2, 3, 4],长度为 4

二、解法分析:从暴力到最优

方法一:暴力法(直观但超时)

思路

对每个数 x,检查 x+1, x+2, ... 是否存在于数组中,记录最长连续序列的长度。

代码
public int longestConsecutive(int[] nums) {int longestStreak = 0;for (int num : nums) {int currentNum = num;int currentStreak = 1;while (arrayContains(nums, currentNum + 1)) {currentNum += 1;currentStreak += 1;}longestStreak = Math.max(longestStreak, currentStreak);}return longestStreak;
}private boolean arrayContains(int[] arr, int num) {for (int i = 0; i < arr.length; i++) {if (arr[i] == num) {return true;}}return false;
}
复杂度
  • 时间复杂度:O(n³)
    • 遍历数组 O(n)
    • 对每个数检查后续 O(n)
    • 每次检查是否存在 O(n)
  • 空间复杂度:O(1)

方法二:排序法(O(n log n),不符合要求但易理解)

思路

先排序数组,再遍历统计连续序列长度。

代码
public int longestConsecutive(int[] nums) {if (nums.length == 0) return 0;Arrays.sort(nums);int longestStreak = 1;int currentStreak = 1;for (int i = 1; i < nums.length; i++) {if (nums[i] != nums[i-1]) {if (nums[i] == nums[i-1] + 1) {currentStreak++;} else {longestStreak = Math.max(longestStreak, currentStreak);currentStreak = 1;}}}return Math.max(longestStreak, currentStreak);
}
复杂度
  • 时间复杂度:O(n log n)(排序主导)
  • 空间复杂度:O(1)(忽略排序的栈空间)

方法三:哈希表优化(O(n),最优解)

思路
  1. 用哈希表存储所有数:快速判断某个数是否存在(O(1))。
  2. 仅从序列起点开始计数:若 x-1 不存在,则 x 是序列起点,开始向后计数。
代码
public int longestConsecutive(int[] nums) {// 用 HashSet 存储所有数,支持 O(1) 查询Set<Integer> numSet = new HashSet<>();for (int num : nums) {numSet.add(num);}int longestStreak = 0;// 遍历每个数,若它是序列起点,则向后计数for (int num : numSet) {// 若 num-1 不存在,则 num 是序列起点if (!numSet.contains(num - 1)) {int currentNum = num;int currentStreak = 1;// 持续检查 currentNum+1 是否存在while (numSet.contains(currentNum + 1)) {currentNum += 1;currentStreak += 1;}longestStreak = Math.max(longestStreak, currentStreak);}}return longestStreak;
}
复杂度
  • 时间复杂度:O(n)
    • 每个数最多被访问两次(一次插入哈希表,一次计数)
  • 空间复杂度:O(n)(哈希表存储所有数)

三、最优解法的核心逻辑

1. 为什么用哈希表?

哈希表(HashSet)提供 O(1) 的查找效率,能快速判断某个数是否存在,避免了暴力法中的嵌套循环。

2. 如何避免重复计数?

关键在于只从序列的起点开始计数。例如,对于序列 [1, 2, 3, 4],我们只在遇到 1 时才开始向后计数,遇到 2, 3, 4 时直接跳过。
判断条件:若 num-1 不存在于哈希表中,则 num 是序列起点。

四、记忆技巧:把代码变成“寻宝游戏”

用场景化记忆法理解最优解法的核心逻辑:

1. 角色赋值

  • 哈希表(numSet):扮演“地图”,标记所有宝藏位置(数字存在)。
  • 遍历过程:扮演“探险家”,在地图上寻找宝藏。
  • 序列起点:扮演“特殊宝藏”,只有找到它才能开启连续挖掘。

2. 游戏规则

① 探险家拿到地图(哈希表),标记所有宝藏位置。
② 探险家随机站在一个数字位置上,检查:

  • 如果左边一格(num-1)没有宝藏,则当前位置是“特殊宝藏”,可开启连续挖掘;
  • 如果左边有宝藏,则跳过当前位置(留给左边的宝藏来挖掘)。
    ③ 挖到一个宝藏后,持续向右挖掘(检查 num+1),记录最长连续挖掘长度。

3. 关键记忆点

  • 只从序列起点开始挖掘 → 避免重复计数。
  • 哈希表快速定位宝藏 → O(1) 查询效率。

五、实战拓展:变种题巩固思路

  1. 允许重复元素:原题已自动处理(哈希表去重)。
  2. 返回具体序列:在计数时记录起点和终点,最后构造序列。
  3. 双向扩展:同时向前(num-1, num-2, ...)和向后扩展,适用于“允许不连续但可排序”的场景。

通过“寻宝游戏”的场景记忆,最优解法的逻辑会变得非常直观。记住:算法的本质是“用合适的数据结构优化操作”,本题中哈希表的作用不仅是存储数据,更是为了快速判断“起点”,从而避免重复计算。多思考这种“筛选起点”的思想,能帮助你解决更多类似问题。

http://www.xdnf.cn/news/15106.html

相关文章:

  • TCP服务器与客户端三种方法实现
  • Linux权限的概念
  • SM712.TCT Semtech TVS二极管——电子设备的终极电路守护
  • DNS(Domain Name System,域名系统)
  • 计算机毕业设计ssm晋中大学城校园论坛 SSM大学城学生社区互动管理平台 JavaWeb高校校园信息交流与服务系统
  • java底层的native和沙箱安全机制
  • 系统思考:多元胜过能力
  • 鸿蒙 Secure Boot 全流程解析:从 BootROM 到内核签名验证的实战指南
  • 2025 年值得尝试的 6 大内容管理系统 (CMS)
  • 【实用IP查询工具】IP数据云-IP地址查询离线库使用方案
  • 【操作系统】Linux 中的 exec 命令
  • RK3566/RK3568 Android11 CAN开发(内核配置+测试验证+安卓app开发)
  • STM32F103之存储/启动流程
  • HarmonyOS基础概念
  • 【TCP/IP】17. 移动 IP
  • Swift 解 LeetCode 324:一步步实现摆动排序 II,掌握数组重排的节奏感
  • 雷达遥感星座微波射频组件抗辐照MCU的选型与实践
  • 【JMeter】接口加密
  • 【JMeter】调试方法
  • 学弟让我帮忙写一个学生管理系统的后端,我直接上科技
  • [大模型问数]实现大模型调用MYSQL(03)【MCP笔记】
  • Webview 中可用的 VS Code 方法
  • Playwright Python 教程:网页自动化
  • 飞算JavaAI:新一代智能编码引擎,革新Java研发范式
  • Linux进程间通信--命名管道
  • 深度学习入门教程(三)- 线性代数教程
  • react打包发到线上报错Minified React error #130
  • 如何快速掌握WeNet:从零到一的端到端语音识别学习指南
  • spring-ai RAG(Retrieval-Augmented Generation)
  • 上位机知识篇---网络通信端口