LeetCode 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 * 10^4
  • -1000 <= numsi <= 1000
  • -10^7 <= k <= 10^7

解题思路

这道题要求找出和为k的连续子数组的个数。我们可以有多种思路:

1. 暴力解法

最直观的方法是枚举所有可能的子数组,计算它们的和,并统计和为k的子数组个数。但这种方法的时间复杂度是O(n²),对于较大规模的数组会超时。

2. 前缀和 + 哈希表

更高效的解法是使用前缀和和哈希表的组合:

  • 使用一个变量 sum 记录从数组起始位置到当前位置的所有元素之和
  • 使用哈希表 prefixSum 记录每种前缀和出现的次数
  • 对于每个位置,我们检查 sum - k 是否在哈希表中存在
    • 如果存在,说明从某个位置到当前位置的子数组和为k
    • 将哈希表中 sum - k 对应的次数加到结果中

这种方法的时间复杂度为O(n),空间复杂度为O(n)。

代码实现

java 复制代码
class Solution {
    public int subarraySum(int[] nums, int k) {
        int count = 0;
        int sum = 0;
        // 哈希表:前缀和 -> 出现次数
        HashMap<Integer, Integer> prefixSum = new HashMap<>();
        
        // 初始化:前缀和为0的情况出现了1次
        prefixSum.put(0, 1);
        
        for (int num : nums) {
            // 累加前缀和
            sum += num;
            
            // 如果 (sum - k) 存在于哈希表中,说明存在若干个前缀和为 (sum - k) 的子数组
            // 这些子数组与当前位置之间的子数组的和为k
            if (prefixSum.containsKey(sum - k)) {
                count += prefixSum.get(sum - k);
            }
            
            // 将当前前缀和加入哈希表,或更新其出现次数
            prefixSum.put(sum, prefixSum.getOrDefault(sum, 0) + 1);
        }
        
        return count;
    }
}

复杂度分析

  • 时间复杂度:O(n),其中 n 是数组的长度。我们只需要遍历一次数组。
  • 空间复杂度:O(n),最坏情况下哈希表需要存储 n 个不同的前缀和。

解题思路详解

这道题的关键在于理解前缀和与哈希表的结合使用。前缀和是指从数组起始位置到当前位置的元素和。

如果我们有两个位置 i 和 j,且它们的前缀和之差等于 k,即 prefixSum[j] - prefixSum[i] = k,那么从位置 i+1 到位置 j 的子数组和就是 k。

转化一下等式:prefixSum[j] - k = prefixSum[i],所以我们只需要知道在位置 j 之前有多少个位置 i 满足 prefixSum[i] = prefixSum[j] - k

使用哈希表记录每个前缀和出现的次数,当我们遍历到位置 j 时,就可以直接查询有多少个位置 i 满足条件,从而在O(1)时间内知道有多少个和为k的子数组以位置 j 结尾。

相关推荐
AI小老六3 小时前
SkillOpt 架构拆解:把 Skill 文本当参数,用执行轨迹训练 Agent
后端·算法·ai编程
胡萝卜术3 小时前
从“分数打架”到“排名投票”:为什么你的ChatBI必须用RRF?
算法·设计模式·面试
Asize4 小时前
初识DFS 与 BFS:递归、队列与图遍历
算法
罗西的思考17 小时前
机器人 / 强化学习】HIL-SERL:人类在环驱动的具身智能进化框架
人工智能·算法·机器学习
CSharp精选营20 小时前
关系型 vs 非关系型:从原理到选型,一文搞定数据库核心分类
数据结构·nosql·关系型数据库·非关系型数据库·技术选型
美团技术团队21 小时前
LongCat 开源 VitaBench 2.0:长期动态智能体基准新标杆
人工智能·算法
To_OC2 天前
LC 207 课程表:刚学图论那会儿,我连这是拓扑排序都没看出来
javascript·算法·leetcode
To_OC2 天前
LC 208 实现 Trie 前缀树:曾被名字劝退,写完发现是送分题
javascript·算法·leetcode
BadBadBad__AK2 天前
线段树维护区间 k 次方和
c++·数学·算法·stl