LeetCode 128. 最长连续序列 | Python

128. 最长连续序列


题目


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

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

示例:数组

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

解题思路


思路:哈希表spa

本题主要的难点在于算法时间复杂度限定为 O(n) 的方法上。code

先假设通常的状况下。能够尝试枚举数组中每一个元素 i,以其起点不断尝试匹配 +1,+2 ... 是否存在于数组中,这样不断枚举并更新最大的长度。固然,这样会很是消耗时间,由于当你枚举 i 结束时,再次遇到元素值 i+1 时,这里又将从新匹配。(考虑改进)blog

能够考虑使用集合的方法,先使用集合,将数组去重。这里须要改进的就是重复匹配的状况。排序

当使用集合的方法时,这里先说明一下,在什么样的状况下才要去开始计算长度,而不须要重复去匹配。若是 i 的前面一个数 i-1 存在于集合当中,那么它在这个时候就须要被跳过。由于它与 i 是能够组成连续序列的,只要在 i-1 的时候计算便可。rem

那么也就是,当 i 存在前驱数 i-1 的时候,不须要计算,直接跳过。只有当 i 无前驱数时,才能够当成是起始的位置开始计算长度。遍历匹配不断更新最大值便可。(具体代码就不贴了,这里给出大概的思路,可尝试编写)get

如今主要介绍本篇幅使用的哈希表的方法。

这里哈希表的键是每一个端点,而值则是它对应连续区间的长度。

具体的作法:

  • 构建哈希表,遍历数组;
  • 若是数已经存在于哈希表中,那么跳过不作处理;
  • 若是数是新数的状况下,那么这里新数将须要查找左右相邻数对应的值(也就是左右相邻数字对应的连续区间的长度)。这里与当前数的长度相加就是这个数对应的新区间长度。length = left + 1 + right
  • 须要与最大的长度值进行比较,更新最大值
  • 同时还须要更新两个端点的区间长度值。

具体的代码实现以下。

代码实现


class Solution:
    def longestConsecutive(self, nums: List[int]) -> int:
        hash_dict = {}

        max_length = 0

        for num in nums:
            # 这里表示新数进来
            # 此时须要查找左右相邻的两个数对应的区间长度
            # 左右两个数的长度 + 自身的长度,就是此时新数对应的区间长度
            if num not in hash_dict:
                # 若是键不在哈希表中,取值 0
                pre_length = hash_dict.get(num - 1, 0)
                next_length = hash_dict.get(num + 1, 0)

                cur_length = pre_length + 1 + next_length
                
                if cur_length > max_length:
                    max_length = cur_length
                
                # 添加新数,同时更新两个端点的值
                # 由于序列是连续的,此时两侧端点对应的区间长度会由于当前数的加入发生改变
                hash_dict[num] = cur_length
                hash_dict[num-pre_length] = cur_length
                hash_dict[num+next_length] = cur_length
        
        return max_length

实现结果


实现结果

总结


  • 本篇幅使用的是哈希表的方法。哈希表的 key 存的是端点,而 value 存储的端点对应的连续区间长度。
  • 具体的实现的作法:

    • 构建哈希表,遍历数组
    • 当遇到存在于哈希表的数字,不作处理跳过
    • 当遇到新数时,须要查找左右相邻的数对应的区间长度,与自身长度相加。(左右数字不存在于哈希表中,返回 0),与最大值比较同时更新最大值。
    • 将新数与其对应的区间长度存入哈希表中,同时更新左右两个端点的值。(例如左边数对应的连续区间长度的起始位置,便是 num - pre_length。num 表示当前数,pre_length 便是左边的连续区间长度,)
    • 最终返回最大值。

文章原创,若是以为写得好,欢迎关注。微信公众号《书所集录》同步更新,一样欢迎关注。
相关文章
相关标签/搜索