子串----和为K的子数组

🔥个人主页: Milestone-里程碑

❄️个人专栏: <<力扣hot100>> <<C++>><<Linux>>

<<Git>><<MySQL>>

🌟心向往之行必能至

题目解读

给定一个整数数组 nums 和一个整数 k,我们需要统计并返回数组中和为 k连续子数组的个数。

这里要注意子数组的定义:它是数组中元素的连续非空序列

示例

  • 输入:nums = [1,1,1], k = 2
  • 输出:2
  • 解释:子数组 [1,1](第 0、1 个元素)和 [1,1](第 1、2 个元素)的和都为 2。

暴力解法:双重循环

最直接的思路是枚举所有可能的子数组,计算它们的和并与 k 比较。

  • 外层循环确定子数组的起点,内层循环确定终点。
  • 时间复杂度:O(n²),在 n = 2 * 10⁴ 的数据规模下会超时。
  • 空间复杂度:O(1)

cpp

复制代码
// 暴力解法(超时)
int subarraySum(vector<int>& nums, int k) {
    int count = 0;
    for (int i = 0; i < nums.size(); ++i) {
        int sum = 0;
        for (int j = i; j < nums.size(); ++j) {
            sum += nums[j];
            if (sum == k) {
                count++;
            }
        }
    }
    return count;
}

优化解法:前缀和 + 哈希表

核心思路

我们定义前缀和 pre_sum[i] 表示数组前 i 个元素的和,那么子数组 nums[j..i] 的和可以表示为:pre_sum[i] - pre_sum[j-1] = k我们的目标就是找到有多少对 (j-1, i) 满足这个等式,等价于找 pre_sum[j-1] = pre_sum[i] - k 的出现次数。

哈希表的作用

我们用一个哈希表 hash 来记录前缀和出现的次数:

  • 初始时,hash[0] = 1,这是为了处理 pre_sum[i] == k 的情况(即子数组从第 0 个元素开始)。
  • 遍历数组时,我们维护当前的前缀和 current_sum,并在哈希表中查找 current_sum - k 的出现次数,累加到结果中。
  • 最后将当前前缀和的出现次数在哈希表中加 1。

代码实现

cpp

复制代码
#include <vector>
#include <unordered_map>

using namespace std;

class Solution {
public:
    int subarraySum(vector<int>& nums, int k) {
        int count = 0;
        int current_sum = 0;
        unordered_map<int, int> hash;
        hash[0] = 1; // 初始化前缀和0出现1次
        
        for (int num : nums) {
            current_sum += num;
            // 如果存在 current_sum - k,说明有子数组的和为k
            if (hash.find(current_sum - k) != hash.end()) {
                count += hash[current_sum - k];
            }
            // 更新当前前缀和的出现次数
            hash[current_sum]++;
        }
        
        return count;
    }
};

复杂度分析

  • 时间复杂度O(n),只需要遍历数组一次,哈希表的查找和插入操作都是 O(1)
  • 空间复杂度O(n),哈希表最多存储 n+1 个前缀和。

关键细节

  1. 初始化哈希表hash[0] = 1 是关键,它处理了子数组从第 0 个元素开始的情况。
  2. 负数和零的处理:由于数组中可能包含负数和零,前缀和可能会重复,哈希表可以很好地统计这些重复的前缀和。
  3. 连续子数组 :前缀和的定义保证了我们统计的是连续的子数组。
相关推荐
小陳参上18 分钟前
用Python创建一个Discord聊天机器人
jvm·数据库·python
2301_7890156219 分钟前
DS进阶:AVL树
开发语言·数据结构·c++·算法
minstbe2 小时前
IC设计私有化AI助手实战:基于Docker+OpenCode+Ollama的数字前端综合增强方案(进阶版)
人工智能·python·语言模型·llama
zyq99101_13 小时前
优化二分查找:前缀和降复杂度
数据结构·python·蓝桥杯
qyzm4 小时前
天梯赛练习(3月13日)
开发语言·数据结构·python·算法·贪心算法
方向研究4 小时前
汽油生产
大数据
逆境不可逃4 小时前
LeetCode 热题 100 之 64. 最小路径和 5. 最长回文子串 1143. 最长公共子序列 72. 编辑距离
算法·leetcode·动态规划
码农小白AI4 小时前
IACheck AI报告文档审核:高端制造合规新助力,保障标准引用报告质量
大数据·人工智能·制造
CoderCodingNo4 小时前
【GESP】C++五级练习题 luogu-P1182 数列分段 Section II
开发语言·c++·算法
放下华子我只抽RuiKe54 小时前
机器学习全景指南-直觉篇——基于距离的 K-近邻 (KNN) 算法
人工智能·gpt·算法·机器学习·语言模型·chatgpt·ai编程