(10)LeetCode 560. 和为K的子数组

LeetCode 560. 和为K的子数组

题目描述

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

示例 1:

复制代码
输入:nums = [1,1,1], k = 2
输出:2

示例 2:

复制代码
输入:nums = [1,2,3], k = 3
输出:2

解题思路

1. 问题转化

假设我们定义前缀和 pre[i] 为数组 nums[0..i] 的和,即
pre[i] = nums[0] + nums[1] + ... + nums[i]

那么,对于任意一个连续子数组 nums[j..i]j ≤ i),它的和可以表示为:
sum(j, i) = pre[i] - pre[j-1](当 j = 0 时,规定 pre[-1] = 0)。

题目要求 sum(j, i) = k,即
pre[i] - pre[j-1] = kpre[j-1] = pre[i] - k

因此,对于每个以 i 结尾的子数组,我们只需要知道在 i 之前 (包括 i 自身吗?注意 j-1 < i,所以是之前)有多少个位置 j-1 的前缀和等于 pre[i] - k。这些位置对应的 j 就是以 i 结尾且和为 k 的子数组的起点。

2. 哈希表优化

我们可以用一个哈希表 hash 来记录遍历过程中每个前缀和出现的次数:

  • 遍历数组,依次计算当前前缀和 cur
  • 对于当前 cur,我们想要知道之前出现过的 cur - k 有多少次,这些次数就是当前 i 结尾的满足条件的子数组个数,累加到答案中。
  • 然后将当前前缀和 cur 的次数加 1,供后续使用。

初始化hash[0] = 1 非常重要。它表示在数组开始之前(即下标 -1 处)有一个虚拟的前缀和为 0。这样当某个 cur == k 时,cur - k = 0 就能从哈希表中取到 1,从而正确统计从下标 0 到 i 的这个子数组。

3. 代码实现细节

原代码将原数组原地修改为前缀和数组,节省了空间。遍历一次并同时更新哈希表和答案。

代码实现(C++)

cpp 复制代码
class Solution {
public:
    int subarraySum(vector<int>& nums, int k) {
        int n = nums.size(), ans = 0;
        // 将原数组转化为前缀和数组(原地修改)
        for(int i = 1; i < n; i++)
            nums[i] += nums[i - 1];

        unordered_map<int, int> hash;  // 记录每个前缀和出现的次数
        hash[0] = 1;                    // 初始化:前缀和为0出现1次(空子数组)

        for(int i = 0; i < n; i++){
            // 当前前缀和减去k,得到需要的前缀和值
            ans += hash[nums[i] - k];
            // 将当前前缀和的计数加1
            hash[nums[i]]++;
        }
        return ans;
    }
};

复杂度分析

  • 时间复杂度O(n),其中 n 是数组的长度。我们只需遍历数组两次(一次计算前缀和,一次遍历统计),哈希表的插入和查询都是 O(1)
  • 空间复杂度O(n),哈希表在最坏情况下需要存储所有不同的前缀和。

总结

本题是前缀和与哈希表结合的经典题目。核心思想是将子数组和问题转化为两个前缀和的差值问题,利用哈希表快速查找之前出现过的前缀和,从而在一次遍历中完成统计。初始化 hash[0] = 1 是处理从数组开头开始的子数组的关键。

拓展

  • 如果题目要求返回所有满足条件的子数组的具体下标,则可以在哈希表中存储前缀和对应的下标列表,然后遍历输出。
  • 如果数组中有负数,前缀和可能不是单调的,但该算法依然适用,因为哈希表只关心值出现的次数,不关心顺序。
相关推荐
Beginner x_u41 分钟前
链表专题:JS 实现原理与高频算法题总结
javascript·算法·链表
_深海凉_4 小时前
LeetCode热题100-寻找两个正序数组的中位数
算法·leetcode·职场和发展
踩坑记录5 小时前
leetcode hot100 寻找两个正序数组的中位数 hard 二分查找 双指针
leetcode
旖-旎5 小时前
深搜练习(电话号码字母组合)(3)
c++·算法·力扣·深度优先遍历
谭欣辰5 小时前
C++快速幂完整实战讲解
算法·决策树·机器学习
Mr_pyx5 小时前
【LeetHOT100】随机链表的复制——Java多解法详解
算法·深度优先
AIFarmer5 小时前
【无标题】
开发语言·c++·算法
AGV算法笔记5 小时前
CVPR 2025 最新感知算法解读:GaussianLSS 如何用 Gaussian Splatting 重构 BEV 表示?
算法·重构·自动驾驶·3d视觉·感知算法·多视角视觉
勤劳的进取家6 小时前
数据链路层基础
网络·学习·算法
Advancer-7 小时前
第二次蓝桥杯总结(上)
java·算法·职场和发展·蓝桥杯