2022-09-05 15:36:37 +08:00
|
|
|
# Leetcode Subarray-Sum-Equals-K
|
|
|
|
|
|
|
|
2022-09-05 15:22
|
|
|
|
|
|
|
|
> ##### Algorithms:
|
|
|
|
>
|
2022-09-06 20:22:48 +08:00
|
|
|
> #algorithm #hash_table
|
2022-09-05 15:36:37 +08:00
|
|
|
>
|
|
|
|
> ##### Data structures:
|
|
|
|
>
|
2022-09-06 20:22:48 +08:00
|
|
|
> #DS #array
|
2022-09-05 15:36:37 +08:00
|
|
|
>
|
|
|
|
> ##### Difficulty:
|
|
|
|
>
|
2022-09-06 20:22:48 +08:00
|
|
|
> #coding_problem #difficulty_medium
|
2022-09-05 15:36:37 +08:00
|
|
|
>
|
|
|
|
> ##### Additional tags:
|
|
|
|
>
|
2022-09-06 20:22:48 +08:00
|
|
|
> #leetcode #CS_list_need_practicing
|
2022-09-05 15:36:37 +08:00
|
|
|
>
|
|
|
|
> ##### Revisions:
|
|
|
|
>
|
|
|
|
> N/A
|
|
|
|
|
|
|
|
##### Links:
|
|
|
|
|
|
|
|
- [Link to problem](https://leetcode.com/problems/subarray-sum-equals-k/)
|
2022-09-05 16:41:34 +08:00
|
|
|
- [Explanation](https://leetcode.com/problems/subarray-sum-equals-k/discuss/102111/Python-Simple-with-Explanation/694092)
|
2022-09-05 15:36:37 +08:00
|
|
|
|
|
|
|
---
|
|
|
|
|
|
|
|
### Problem
|
|
|
|
|
|
|
|
Given an array of integers `nums` and an integer `k`, return _the total number of subarrays whose sum equals to_ `k`.
|
|
|
|
|
|
|
|
A subarray is a contiguous **non-empty** sequence of elements within an array.
|
|
|
|
|
|
|
|
#### Examples
|
|
|
|
|
|
|
|
**Example 1:**
|
|
|
|
|
|
|
|
**Input:** nums = [1,1,1], k = 2
|
|
|
|
**Output:** 2
|
|
|
|
|
|
|
|
**Example 2:**
|
|
|
|
|
|
|
|
**Input:** nums = [1,2,3], k = 3
|
|
|
|
**Output:** 2
|
|
|
|
|
|
|
|
#### Constraints
|
|
|
|
|
2022-09-06 20:22:48 +08:00
|
|
|
- `1 <= nums.length <= 2 * 104`
|
|
|
|
- `-1000 <= nums[i] <= 1000`
|
|
|
|
- `-107 <= k <= 107`
|
2022-09-05 15:36:37 +08:00
|
|
|
|
|
|
|
### Thoughts
|
|
|
|
|
|
|
|
> [!summary]
|
2022-09-06 20:22:48 +08:00
|
|
|
> This can be solved using #prefix_sum and #hash_table
|
2022-09-05 15:36:37 +08:00
|
|
|
|
2022-09-06 20:22:48 +08:00
|
|
|
> I over-complicated the solution by adding stuff like
|
2022-09-05 15:36:37 +08:00
|
|
|
> sorting, but it turn out to be not so difficult.
|
|
|
|
|
2022-09-05 16:10:30 +08:00
|
|
|
> [!tip]
|
2022-09-06 20:22:48 +08:00
|
|
|
> Tried using sliding window, but it doesn't work because of
|
2022-09-05 16:10:30 +08:00
|
|
|
> negative numbers.
|
|
|
|
|
2022-09-06 20:22:48 +08:00
|
|
|
This one is harder to understand, making it a medium
|
2022-09-05 16:41:34 +08:00
|
|
|
difficulty problem.
|
2022-09-05 16:10:30 +08:00
|
|
|
|
2022-09-05 16:41:34 +08:00
|
|
|
The intuition is using prefix to count the subarray sum, and
|
2022-09-06 20:22:48 +08:00
|
|
|
use hashmaps to count the **number of subarray sums**. This
|
|
|
|
takes care of **negative numbers** which is not solvable
|
2022-09-05 16:41:34 +08:00
|
|
|
using sliding windows.
|
|
|
|
|
|
|
|
We step one cell at a time, and try to ask ourselves:
|
|
|
|
`Can we hop back K to land on a cell?`
|
|
|
|
|
|
|
|
And we answer this question by using hash maps and recording
|
|
|
|
the data on the fly.
|
2022-09-05 15:36:37 +08:00
|
|
|
|
|
|
|
### Solution
|
|
|
|
|
2022-09-05 16:41:34 +08:00
|
|
|
```cpp
|
|
|
|
class Solution {
|
|
|
|
public:
|
|
|
|
int subarraySum(vector<int> &nums, int k) {
|
|
|
|
unordered_map<int, int> umap;
|
|
|
|
// sum == k is a solution, should be counted
|
|
|
|
umap[0] = 1;
|
|
|
|
|
|
|
|
int sum = 0;
|
|
|
|
int count = 0;
|
|
|
|
|
|
|
|
for (int i : nums) {
|
|
|
|
// update the prefix sum
|
|
|
|
sum += i;
|
|
|
|
|
|
|
|
if (umap.find(sum - k) != umap.end()) {
|
|
|
|
// we can land on a cell, then add the counter.
|
|
|
|
count += umap[sum - k];
|
|
|
|
}
|
|
|
|
|
|
|
|
// record this step.
|
|
|
|
umap[sum]++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
```
|
2022-09-05 15:36:37 +08:00
|
|
|
|
|
|
|
TLE, brute force
|
2022-09-06 20:22:48 +08:00
|
|
|
|
2022-09-05 15:36:37 +08:00
|
|
|
```cpp
|
|
|
|
class Solution {
|
|
|
|
public:
|
|
|
|
int subarraySum(vector<int> &nums, int k) {
|
|
|
|
// O(N^2 / 2)
|
|
|
|
|
|
|
|
int n = nums.size();
|
|
|
|
|
|
|
|
if (n == 1) {
|
|
|
|
return (k == nums[0]);
|
|
|
|
}
|
|
|
|
|
|
|
|
vector<int> prefix(n + 1, 0);
|
|
|
|
int prefixSum = 0;
|
|
|
|
|
|
|
|
for (int i = 0; i < n; i++) {
|
|
|
|
prefixSum += nums[i];
|
|
|
|
prefix[i + 1] = prefixSum;
|
|
|
|
}
|
|
|
|
|
|
|
|
int count = 0;
|
|
|
|
|
|
|
|
for (int i = 0; i < n + 1; i++) {
|
|
|
|
for (int j = i + 1; j < n + 1; j++) {
|
|
|
|
if (prefix[j] - prefix[i] == k) {
|
|
|
|
count++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
};
|
2022-09-05 16:10:30 +08:00
|
|
|
```
|