【中等】560. 和为 K 的子数组

题目

560. 和为 K 的子数组

给你一个整数数组 nums 和一个整数 k ,请你统计并返回 该数组中和为 k 的子数组的个数 。

子数组是数组中元素的连续非空序列。

示例 1:

输入:nums = [1,1,1], k = 2
输出:2
示例 2:

输入:nums = [1,2,3], k = 3
输出:2

提示:

1 <= nums.length <= 2 * 104
-1000 <= nums[i] <= 1000
-107 <= k <= 107


题解

  1. 最基本的,暴力解法
    两个边界,一定一遍历,需要O(n ^ 2);求和计算还需要 O(n);
    最终时间复杂度O(n^3)。
    超时。

  2. 优化1:对内部循环进行优化
    求和计算与遍历过程可以综合优化。O(n)的求和计算可以改写为 O(1)过程。
    优化后相当于只在乎尾部的遍历,需要O(n)计算量;
    对头部指针的遍历和求和计算可以放在一起,每次遍历时求和,需要O(n)计算量;
    优化后时间复杂度为O(n^2)。

  • 不能一直是头不动尾动的思想
  1. 优化2:前缀和+哈希表

通过计算前缀和,将问题转化为求解两个前缀和之差等于k的情况。遍历一次,就可以统计出和为k的连续子数组的个数。时间复杂度为O(n),空间复杂度为O(n)。

假设数组的前缀和数组为prefixSum,其中prefixSum[i]表示从数组起始位置到第i个位置的元素之和。那么对于任意的两个下标i和j(i < j),如果prefixSum[j] - prefixSum[i] = k,即从第i个位置到第j个位置的元素之和等于k,那么说明从第i+1个位置到第j个位置的连续子数组的和为k。

通过遍历数组,计算每个位置的前缀和,并使用一个哈希表来存储每个前缀和出现的次数。在遍历的过程中,我们检查是否存在prefixSum[j] - k的前缀和,如果存在,说明从某个位置到当前位置的连续子数组的和为k,我们将对应的次数累加到结果中。

前缀和第一个值规定为0,例如以下情况:

nums = [1, 1], k=2

class Solution:
    def subarraySum(self, nums: List[int], k: int) -> int:
        # 要求的连续子数组
        count = 0
        n = len(nums)
        preSums = collections.defaultdict(int)
        preSums[0] = 1

        presum = 0
        for i in range(n):
            presum += nums[i]
            
            # if preSums[presum - k] != 0:
            count += preSums[presum - k]   # 利用defaultdict的特性,当presum-k不存在时,返回的是0。这样避免了判断

            preSums[presum] += 1  # 给前缀和为presum的个数加1
            
        return count

作者:无穷升高的卡农
链接:https://leetcode.cn/problems/subarray-sum-equals-k/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

  1. 目前Python两种方法都会超时。前缀和方法84超时。

解题

  1. 中等题没试暴力。
  2. 枚举优化
class Solution(object):
    def subarraySum(self, nums, k):
        """
        :type nums: List[int]
        :type k: int
        :rtype: int
        """
        result = 0

        for i in range(len(nums)): 
            tail = i
            head = tail
            t = 0
            while head >= 0:
                t += nums[head]
                if t == k:
                    result+=1
                head -= 1

        return result


  1. 前缀和方法
class Solution(object):
    def subarraySum(self, nums, k):
        """
        :type nums: List[int]
        :type k: int
        :rtype: int
        """
        presum = {}
        presum[0] = 1
        result = 0
        s = 0
        for i in nums:
            s += i
            if s-k in presum.keys():
                result+=presum[s-k]
            if s in presum.keys():
                presum[s] += 1
            else:
                presum[s] = 1
        return result
        

问题

  • collections.defaultdict的用法,免去了很多判断,代码比较简洁。
  • 以前都是定前动后,现在动后定前,需要打破思维惯性。
  • 前缀和方法
  • 前缀和初值为0
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
请用C++解决这个leetcode的问题:2653. 滑动数组的美丽值 提示 中等 26 相关企业 给你一个长度为 n 的整数数组 nums ,请你求出每个长度为 k 的数组的 美丽值 。 一个数组的 美丽值 定义为:如果数组中第 x 小整数 是 负数 ,那么美丽值为第 x 小的数,否则美丽值为 0 。 请你返回一个包含 n - k + 1 个整数的数组,依次 表示数组中从第一个下标开始,每个长度为 k 的数组的 美丽值 。 数组指的是数组中一段连续 非空 的元素序列。 示例 1: 输入:nums = [1,-1,-3,-2,3], k = 3, x = 2 输出:[-1,-2,-2] 解释:总共有 3 个 k = 3 的数组。 第一个数组是 [1, -1, -3] ,第二小的数是负数 -1 。 第二个数组是 [-1, -3, -2] ,第二小的数是负数 -2 。 第三个数组是 [-3, -2, 3] ,第二小的数是负数 -2 。 示例 2: 输入:nums = [-1,-2,-3,-4,-5], k = 2, x = 2 输出:[-1,-2,-3,-4] 解释:总共有 4 个 k = 2 的数组。 [-1, -2] 中第二小的数是负数 -1 。 [-2, -3] 中第二小的数是负数 -2 。 [-3, -4] 中第二小的数是负数 -3 。 [-4, -5] 中第二小的数是负数 -4 。 示例 3: 输入:nums = [-3,1,2,-3,0,-3], k = 2, x = 1 输出:[-3,0,-3,-3,-3] 解释:总共有 5 个 k = 2 的数组。 [-3, 1] 中最小的数是负数 -3 。 [1, 2] 中最小的数不是负数,所以美丽值为 0 。 [2, -3] 中最小的数是负数 -3 。 [-3, 0] 中最小的数是负数 -3 。 [0, -3] 中最小的数是负数 -3 。 提示: n == nums.length 1 <= n <= 105 1 <= k <= n 1 <= x <= k -50 <= nums[i] <= 50
05-24

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值