算法题-16

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

子数组是数组中元素的连续非空序列。

示例 1:

输入:nums = [1,1,1], k = 2

输出:2

示例 2:

输入:nums = [1,2,3], k = 3

输出:2

对于这道题如果我们一个一个试,我们也会试出来答案是2

复制代码
[1]        = 1 ❌
[1,1]      = 2 ✅
[1,1,1]    = 3 ❌
   [1]     = 1 ❌
   [1,1]   = 2 ✅
      [1]  = 1 ❌

但是我们在试的是够的一个关键问题是从某个位置j到i的连续和,nums[j] + ... + nums[i]。

假如我们一直往右边加,我们会发现nums = [1,1,1],这个就是前缀和

位置 当前累计和
开始 0
0 1
1 2
2 3

看这个表,如果我们想知道有没有一段子数组和=2,我们发下,当我们在2位置的时候,前面有一个时刻的和是1,那么3-1=2,说明中间的那段=2,这样我们就找到了一个子数组

时间线:

0 ---- 1 ---- 2 ---- 3

↑ ↑

旧位置 现在

所以整体唯一的思想就是,不断地询问以前有没有一个"旧的累计和",让:当前和 - 旧和 = k,也就是,旧和 = 当前和 - k

所以代码的思路是:

初始化map = { 0:1 }sum = 0,count = 0

为什么有 0?表示:还没开始时,和就是 0。

第一步:遇到 1,sum = 1,我们找:sum - k = 1 - 2 = -1,map 里没有,记录:map = {0:1, 1:1}

第二步:遇到 1,sum = 2,找:2 - 2 = 0,map 里有,说明:从开头到这里 = 2,count++,count = 1,记录:map = {0:1,1:1,2:1}

第三步:遇到 1,sum = 3,找:3 - 2 = 1,map 里有 ,说明:中间一段 = 2,count++,count = 2

最终答案2

javascript 复制代码
function subarraySum(nums, k) {
  const map = new Map();
  map.set(0, 1);

  let sum = 0;
  let count = 0;

  for (let num of nums) {
    sum += num;

    // 找过去的某个和
    if (map.has(sum - k)) {
      count += map.get(sum - k);
    }

    // 记录现在
    map.set(sum, (map.get(sum) || 0) + 1);
  }

  return count;
}
相关推荐
故事和你911 天前
sdut-程序设计基础Ⅰ-实验五一维数组(8-13)
开发语言·数据结构·c++·算法·蓝桥杯·图论·类和对象
像污秽一样1 天前
算法与设计与分析-习题4.2
算法·排序算法·深度优先·dfs·bfs
Storynone1 天前
【Day20】LeetCode:39. 组合总和,40. 组合总和II,131. 分割回文串
python·算法·leetcode
明明如月学长1 天前
AI 更新太快学不过来?我用OpenClaw打造专属AI学习工作流
算法
黎阳之光1 天前
【黎阳之光:以无线专网与视频孪生,赋能智慧广电与数字中国】
算法·安全·智慧城市·数字孪生
刀法如飞1 天前
Agentic AI时代,程序员必备的算法思想指南
人工智能·算法·agent
刀法如飞1 天前
Agentic AI时代程序员必备算法思想详解(附实战案例)
算法·ai编程·编程开发·agentic
飞Link1 天前
告别盲目找Bug:深度解析 TSTD 异常检测中的预测模型(Python 实战版)
开发语言·python·算法·bug
记忆多1 天前
c++名字空间 函数模版 左右值
开发语言·c++·算法
三伏5221 天前
控制理论前置知识——相平面数学基础2(示例部分)
算法·平面·控制