(双指针)LeetCode 209 长度最小的子数组

灵神题单:据说很好的题单,mark一下

分享丨【算法题单】滑动窗口与双指针(定长/不定长/单序列/双序列/三指针/分组循环) - 讨论 - 力扣(LeetCode)

LeetCode 209 长度最小的子数组

问题分析:

给定一个只包含正数的数组和一个目标值 `target`,我们需要找到和大于等于 `target` 的最短子数组长度。

暴力解法:

最直接的方法是 固定左端点,枚举右端点,并在过程中计算子数组和:

  • 以第 0 个元素为左端点时,枚举子数组 `a, ab, abc, abcd, abcde`;
  • 以第 1 个元素为左端点时,枚举 `b, bc, bcd, bcde`;
  • 依此类推直到末尾。

每个子数组求和后,若 sum ≥ target,就用当前长度更新最小值 ans。

时间复杂度:O(n²)。

第一次优化:

由于数组中元素都是正数,如果从某个左端点开始,当累加到 `a+b+c < target`,继续加 `d` 后得到 `a+b+c+d ≥ target`,那么再加 `e` 一定也满足 `≥ target`。因此,当首次达到 `≥ target` 时就可以 提前 break,不需要继续枚举更长的子数组。

时间复杂度仍为 O(n²),但减少了部分冗余。

最终优化(滑动窗口 / 双指针):

进一步利用"正数数组"的 **单调性**。

  • 假设从左端点 `a` 开始,`a+b+c < target` 且 `a+b+c+d ≥ target`,则以 `a` 为起点的最短子数组长度为 4。
  • 接着考虑左端点为 `b`,由于 `a+b+c < target`,则 `b` 和 `b+c` 一定也 `< target`,因此无需从 b 重新累加,可以直接从上一次窗口 `a+b+c+d` 的和中减去 `a` 继续。
  • 如果 `b+c+d ≥ target`,则以 `b` 为起点的最短子数组长度为 3;否则继续扩展右指针,直到没有新元素可加入。

这样每个元素最多被访问两次(一次进窗口,一次出窗口),时间复杂度优化到 O(n)。

这个版本有的会超时,可能因为sum那里的复杂度问题

python 复制代码
class Solution(object):
    def minSubArrayLen(self, target, nums):
        """
        :type target: int
        :type nums: List[int]
        :rtype: int
        """
        length=float('inf')
        slow,fast=0,0
        while fast<len(nums):
            while(sum(nums[slow:fast+1])<target):
                fast=fast+1
                if fast==len(nums):
                    return 0 if length==float('inf') else length
            length=min(length,fast-slow+1)
            slow=slow+1
        if length==float('inf'):
            return 0
        else:
            return length
        

这个可以AC,total+=nums[fast]的时候要注意先return再加,否则可能边界超出

python 复制代码
class Solution(object):
    def minSubArrayLen(self, target, nums):
        """
        :type target: int
        :type nums: List[int]
        :rtype: int
        """
        length=float('inf')
        slow,fast=0,0
        total=nums[fast]
        while fast<len(nums):
            while(total<target):
                fast=fast+1
                if fast==len(nums):
                    return 0 if length==float('inf') else length
                total+=nums[fast]
            total-=nums[slow]
            length=min(length,fast-slow+1)
            slow=slow+1
        if length==float('inf'):
            return 0
        else:
            return length
相关推荐
前端小白在前进几秒前
力扣刷题:复原IP地址
tcp/ip·算法·leetcode
yaoh.wang13 分钟前
力扣(LeetCode) 94: 二叉树的中序遍历 - 解法思路
python·算法·leetcode·面试·职场和发展·二叉树·跳槽
Evand J14 分钟前
【课题推荐】基于视觉(像素坐标)与 IMU 的目标/自身运动估计(Visual-Inertial Odometry, VIO),课题介绍与算法示例
人工智能·算法·计算机视觉
xu_yule15 分钟前
算法基础(背包问题)—分组背包和混合背包
c++·算法·动态规划·分组背包·混合背包
蓝色汪洋22 分钟前
数码串和oj
数据结构·算法
资深web全栈开发24 分钟前
并查集(Union-Find)套路详解
leetcode·golang·并查集·unionfind
努力学算法的蒟蒻25 分钟前
day39(12.20)——leetcode面试经典150
算法·leetcode·面试
科学最TOP34 分钟前
xLSTM-Mixer:基于记忆混合的多变量时间序列预测
大数据·人工智能·算法·机器学习·时间序列
xlq223221 小时前
29.哈希(下)
算法·哈希算法·散列表
阿昭L1 小时前
leetcode链表是否有环
算法·leetcode·链表