哈希与前缀和

一、一句话理解"前缀和"

不要把数组看成一个个数字,把它看成你在公路上走路

"前缀和"就是你此时此刻脚下踩着的里程碑坐标(距离起点的总路程)。

想求中间某一段连续的路有多长?很简单:当前里程碑 - 过去某个历史里程碑

二、什么时候用?

1. 题目描述标志(题眼)

  • 明确要求找 "连续子数组""连续子串"

  • 明确要求求 "和为 K""和能被 K 整除"

  • 变种描述:求 "包含相同数量的 0 和 1" 的最长连续子数组(把 0 视作 -1,问题瞬间变成求"和为 0"的子数组)。

2. 数据特征标志(绝对死穴)

  • 数组中包含负数! 这是最强烈的信号。如果数组全是正数,优先用"滑动窗口";一旦出现负数,滑动窗口直接瘫痪,必须上"前缀和"。

3. 复杂度要求

  • 题目数组很长(如 10510^5105),要求 O(n)O(n)O(n) 时间复杂度,不许用双层 for 循环暴力解。

三、核心通关公式

每次往前走一步,算出一个"当前总和(cursum)",然后立刻问自己一个问题:

我要找的历史起点 = 当前总和 (cursum) - 目标和 (K)

算出这个"历史起点"后,去你的哈希表(记录本)里查一下,以前有没有经过这个点?如果有,直接把次数加到结果里!

四、哈希表 (Map) 在这充当什么角色?

它是你的"历史坐标记录本"。

  • 存什么{ 历史前缀和 : 这个前缀和出现过的次数 }

  • 为什么存次数:因为数组里有负数,你可能会"进两步退一步",导致同一个里程碑被你踩过好几次。每一次踩过,都能连出一条新的合法路线。

五、极其致命的防坑细节

写循环前,永远记得写上第一行:map.set(0, 1);

  • 大白话解释:在还没迈出第一步之前,你在"0公里"的起点处站了 1 次。

  • 为什么 :如果有个合法的子数组刚好是从数组开头第一个数字算起的,如果你不把 0 提前存进去,这个答案就会被漏掉!

六、肌肉记忆:通用代码模板

这是一个几乎可以套用所有"求连续子数组和"的万能骨架:

javascript 复制代码
var subarraySumTemplate = function(nums, k) {
    let res = 0;
    let cursum = 0;         // 记录当前的里程碑坐标
    let map = new Map();    // 历史坐标记录本

    // 致命细节:初始化起点,防止漏掉从头开始的合法子数组
    map.set(0, 1);

    for (let i = 0; i
相关推荐
罗西的思考9 小时前
机器人 / 强化学习】HIL-SERL:人类在环驱动的具身智能进化框架
人工智能·算法·机器学习
美团技术团队12 小时前
LongCat 开源 VitaBench 2.0:长期动态智能体基准新标杆
人工智能·算法
To_OC1 天前
LC 207 课程表:刚学图论那会儿,我连这是拓扑排序都没看出来
javascript·算法·leetcode
To_OC1 天前
LC 208 实现 Trie 前缀树:曾被名字劝退,写完发现是送分题
javascript·算法·leetcode
BadBadBad__AK1 天前
线段树维护区间 k 次方和
c++·数学·算法·stl
_清歌2 天前
DSpark 深度解读:DeepSeek-V4 如何用「半自回归」把推理速度提升 85%
算法
统计实现局2 天前
SVD 的三步走:双对角化、Givens 收敛、排序
算法
躬行见万象2 天前
《VLA 系列》UniLab 强化训练 | G1 机器人 |复现
算法
统计实现局2 天前
对称不定分解(Bunch-Kaufman):为什么 Cholesky 不够用
算法