和为 k 的子数组
Tips
题目类型: 前缀和 + HashMap
相关题目:
题目
给定一个整数数组和一个整数 k, 你需要找到该数组中和为 k 的连续的子数组的个数.
示例
输入: nums = [1, 1, 1], k = 2
输出: 2
解释: [1, 1] 与 [1, 1] 为两种不同的情况.
输入: nums = [1, 2, 3], k = 3
输出: 2
解释: 结果为 [1, 2] 与 [3]
题解
- 暴力法 🐸
- 暴力前缀和 🐸
- 前缀和 + HashMap
暴力法核心就是整个双循环穷尽所有子数组, 如果这个子数组的和等于 k, 那就让 count++, 直到循环完毕.
var subarraySum = function (nums, k) {
const len = nums.length
let count = 0
for (let i = 0; i < len; ++i) {
let sum = 0
for (let j = i; j < len; ++j) {
sum += nums[j]
if (sum === k) count++
}
}
return count
}
这道题的核心是前缀和, 如果我们想求 nums[i]
到 nums[j]
的和, 可以使用 preSum[j + 1] - preSum[i]
得出.
但在提交时提示超时, 原因就是穷尽所有子数组时整了个双循环, 时间复杂度仍然是 O(n²).
/**
* @param {number[]} nums
* @param {number} k
* @return {number}
*/
var subarraySum = function (nums, k) {
const len = nums.length
const preSum = new Array(len + 1).fill(0)
// 构造前缀和
// num = [3, 5, 2, -2, 4, 1]
for (let i = 0; i < len; i++) {
preSum[i + 1] = preSum[i] + nums[i] // [0, 3, 8, 10, 8, 12, 13]
}
let count = 0
// 穷举所有子数组
for (let left = 0; left < len; left++) {
for (let right = left; right < len; right++) {
// 利用前缀和的性质
if (preSum[right + 1] - preSum[left] === k) {
count++
}
}
}
return count
}
- 设
pre[i]
为pre[0..i]
所有元素的和, 那么pre[i] = pre[i - 1] + nums[i]
, 因为pre[i]
可以由pre[i - 1]
递推过来 - 同理,
pre[j..i]
这个子数组和为 k 这个条件可以转化为pre[i] = pre[j - 1] + k
- 上面的公式也就是
pre[j - 1] = pre[i] - k
上面这个图就很清晰, presum - k
其实就是子串的和, 我们的目的是计算 pre[i] - k
是否等于 presum - k
. 换句话说, 就是在总和里减去 k 是否存在, 如果存在, 那么这就是一个和为 k 的子数组. 因此我们可以通过建立一个 hash 表, 将 pre[i] - k
设为 key, 而这个 key 出现的次数作为 val, 而我们最终的目的是统计 val.
/**
* @param {number[]} nums
* @param {number} k
* @return {number}
*/
var subarraySum = function (nums, k) {
// 前缀和要初始化一个 key 为 0, 因为初始化了一个 0, 所以它出现的次数就是 1
const map = new Map([[0, 1]])
let preSum = 0
let count = 0
for (const num of nums) {
preSum += num
if (map.has(preSum - k)) {
count += map.get(preSum - k)
}
map.set(preSum, map.has(preSum) ? map.get(preSum) + 1 : 1)
}
return count
}