LeetCode实战128 最长连续序列

原题链接

题目描述

给定一个未排序的整数数组,找出最长连续序列的长度。

要求算法的时间复杂度为O(n)

示例:

输入: [100, 4, 200, 1, 3, 2]
输出: 4
解释: 最长连续序列是 [1, 2, 3, 4]。它的长度为 4。

题目解析

乍一看,我们能马上想到的方法,就是先进行一次排序,再查找,可是就算我们常用的快速排序也有O(nlogn)的复杂度,不合题意,所以这道题我们不能直接用排序。

这时,我们可以牺牲一些空间,来降低时间复杂度。

解法:哈希表

将无序变为有序,并不一定需要直接排序,我们可以用哈希表来“曲线救国”。虽然题目给出的序列是无序的,但是哈希表可以非常高效地访问无序序列。

我们可以建立一个无序哈希表unordered_map<int, bool> visited,哈希表的键对应数组的索引,其值表示数组中的元素是否被访问过。

对于每个元素,我们可以以该元素为中心,分别向左、向右查找连续元素,直到不连续为止,记录这个序列的长度,直到找出最大长度。

遍历每个元素,再以每个元素为中心再次进行迭代,似乎,这是一个O(n^2)的算法呀?

别担心!在遍历每个中心元素时,我们会把标记为true的元素跳过,因为,在这个题目的条件下,一个元素只能属于一个连续序列 ,所以我们无需再关心被访问过的中心元素,它一定会被某一个连续序列所包含。我们的算法,仍然属于O(n)级别。

代码:

class Solution {
public:
    int longestConsecutive(vector<int>& nums) {
        unordered_map<int, bool> visited;
        for(auto i : nums)
            visited[i] = false;
        int maxLen = 0;
        
        for(auto i : nums){
            if(visited[i])continue;//如果这个元素被访问过,那么他一定就不再属于另外一个连续序列,直接跳过
            
            int len = 1;//当前序列的长度
            visited[i] = true;
            for(int j = i + 1; visited.find(j) != visited.end(); ++j){
                visited[j] = true;
                len++;
            }
            for(int j = i - 1; visited.find(j) != visited.end(); --j){
                visited[j] = true;
                len++;
            }
            maxLen = max(maxLen, len);
        }
        return maxLen;
    }
};

复杂度分析

时间复杂度:O(n),请仔细思考上面的解题思路

空间复杂度:O(n),因为引入了一个哈希表

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 一些概念 数据结构就是研究数据的逻辑结构和物理结构以及它们之间相互关系,并对这种结构定义相应的运算,而且确保经过这...
    Winterfell_Z阅读 5,979评论 0 13
  • --- layout: post title: "如果有人问你关系型数据库的原理,叫他看这篇文章(转)" date...
    蓝坠星阅读 824评论 0 3
  • 1)这本书为什么值得看: Python语言描述,如果学的Python用这本书学数据结构更合适 2016年出版,内容...
    孙怀阔阅读 12,624评论 0 15
  • 树(续) 二叉树 二叉排序树 二叉排序树,又叫二叉查找树,它或者是一棵空树;或者是具有以下性质的二叉树: 若它的左...
    liuzhangjie阅读 1,167评论 0 0
  • LeetCode 刷题随手记 - 第一部分 前 256 题(非会员),仅算法题,的吐槽 https://leetc...
    蕾娜漢默阅读 17,934评论 2 36