LeetCode209-长度最小的子数组

三月的第一天

我希望我的文章能出现在校刊上

哇卡卡卡卡卡卡卡


题目描述:

给定一个含有 n 个正整数的数组和一个正整数 s ,找出该数组中满足其和 ≥ s 的长度最小的连续子数组。如果不存在符合条件的连续子数组,返回 0。

示例:

输入: s = 7, nums = [2,3,1,2,4,3]
输出: 2
解释: 子数组 [4,3] 是该条件下的长度最小的连续子数组。

解题思路:

这一题我用的是双指针法,无非是固定一个start, end指针。end指针负责右移扩展数组,start指针也是右移不过是精简数组。简言之就是如果nums[start:end]子数组之和大于或等于s,那么end指针不动,start指针右移;如果nums[start:end]子数组之和小于s,那么start指针不动,end指针右移。直至end遍历完了数组,结束遍历取出最小连续子数组的长度即可。

代码如下:

class Solution(object):
    def minSubArrayLen(self, s, nums):
        """
        :type s: int
        :type nums: List[int]
        :rtype: int
        """
        start, end = 0, 1
        flag = False
        min_length = len(nums)
        while start < end:
            all_sum = sum(nums[start:end])
            if all_sum >= s:
                min_length = min(min_length, end-start)
                start += 1
                flag = True
            else:
                end += 1
                if end > len(nums):
                    break
        return min_length if flag is True else 0


if __name__ == "__main__":
    s = 3
    nums = [1, 1]
    min_length = Solution().minSubArrayLen(s, nums)
    print(min_length)

很奇怪的是,执行效率极其的差,耗时1500ms,这简直不能忍嘛!

于是我查看了那些耗时只有几十ms的解法,感觉大同小异,都是用双指针法,个人觉得外层循环用for更快吧,我找了个解法如下,大家可以比较看下差异。

28ms代码如下:

class Solution(object):
    def minSubArrayLen(self, s, nums):
        """
        :type s: int
        :type nums: List[int]
        :rtype: int
        """
        l = 0
        tmp = 0
        res = float('inf')
        for r in range(len(nums)):
            tmp += nums[r]
            while tmp >= s:
                res = min(r-l+1, res)
                tmp-=nums[l]
                l+=1
        return res if res != float('inf') else 0


if __name__ == "__main__":
    s = 3
    nums = [1, 1]
    min_length = Solution().minSubArrayLen(s, nums)
    print(min_length)

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

学习的学习者

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值