hot100 128.最长连续序列

思路:

1.题目要求时间复杂度为O(n),而排序的时间复杂度是O(nlogn),因此本题不能排序。

2.核心思路:对于nums中的元素x,以x为起点,不断查找下一个数x + 1,x + 2,...是否在nums中,并统计序列的长度。

3.为了做到O(n)的时间复杂度,需要做到两个关键优化。

(1)把nums中的数都放到一个哈希集合中,这样可以以O(1)的时间复杂度判断数字是否在nums中。

(2)如果x - 1在哈希集合中,则不以x为起点。这是因为以x - 1为起点计算出的序列长度,一定要比以x为起点计算出的序列长度要长,这样可以避免大量重复计算。比如nums == [3,2,4,5],从3开始,可以找到3,4,5这个连续序列;而从2开始,则可以找到2,3,4,5这个连续序列,一定比从3开始找到的连续序列要长。

4.注意:遍历元素的时候,要遍历哈希集合,而不是nums。如果nums =[1,1,1,...,1,2,3,4,5,...](前一半都是1),遍历nums的做法会导致每个1都跑一个O(n)的循环,总的循环次数是O(n^2),会超时。

附代码:

java 复制代码
class Solution {
    public int longestConsecutive(int[] nums) {
        Set<Integer> set = new HashSet<>();
        for(int num : nums){
            set.add(num); //把nums转换成哈希集合
        }
        int ans = 0;
        for(int x : set){ //遍历哈希集合
            if(set.contains(x - 1)){
                //如果x不是序列的起点,则直接跳过
                continue;
            }
            //x是序列的起点
            int y = x + 1;
            while(set.contains(y)){ //不断查找下一个数是否在哈希集合中
                y++;
            }
            // 循环结束后,y - 1就是最后一个在哈希集合中的数
            // 长度为 y - 1 - x + 1 = y - x
            ans = Math.max(ans,y - x);
        }
        return ans;
    }
}

小优化:设m为nums中不同元素的个数(即哈希集合的大小)。各个连续序列(链)是相互独立的,如果发现其中一条链的长度至少为m/2(长度×2>=m),由于不可能还有一条长度大于m/2的链(否则这两条链的长度之和就超过m了),答案不会再增大,此时可以直接返回答案。

java 复制代码
class Solution {
    public int longestConsecutive(int[] nums) {
        Set<Integer> set = new HashSet<>();
        for(int num : nums){
            set.add(num); //把nums转换成哈希集合
        }
        int m = set.size();
        int ans = 0;
        for(int x : set){ //遍历哈希集合
            if(set.contains(x - 1)){
                //如果x不是序列的起点,则直接跳过
                continue;
            }
            //x是序列的起点
            int y = x + 1;
            while(set.contains(y)){ //不断查找下一个数是否在哈希集合中
                y++;
            }
            // 循环结束后,y - 1就是最后一个在哈希集合中的数
            // 长度为 y - 1 - x + 1 = y - x
            ans = Math.max(ans,y - x);
            if(ans * 2 >= m){
                break;
            }
        }
        return ans;
    }
}
相关推荐
elseif12311 小时前
CSP-S提高级大纲
开发语言·数据结构·c++·笔记·算法·大纲·考纲
Book思议-11 小时前
【数据结构实战】双向链表:在指定位置插入数据
c语言·数据结构·算法·链表
白昼流星!12 小时前
顺序表与单链表的数据存储差异: 为何顺序表元素用指针,链表节点数据不用?
数据结构·链表·顺序表
y = xⁿ13 小时前
【LeetCodehot100】T108:将有序数组转换为二叉搜索树 T98:验证搜索二叉树
数据结构·算法·leetcode
不是株13 小时前
算 法
数据结构·python·算法
自信1504130575913 小时前
插入排序算法
c语言·数据结构·算法·排序算法
阿Y加油吧13 小时前
力扣打卡day09——缺失的第一个正数、矩阵置零
数据结构·算法·leetcode
仰泳的熊猫13 小时前
题目2576:蓝桥杯2020年第十一届省赛真题-解码
数据结构·c++·算法·蓝桥杯
灰色小旋风14 小时前
力扣16 最接近的三数之和(C++)
数据结构·c++·算法·leetcode
前端达人14 小时前
第 4 篇:内容即数据——frontmatter 规范、数据结构与构建链路的工程化设计
大数据·数据结构