news 2026/4/17 21:05:10

双指针专题(十):恰好等于的困境——「K 个不同整数的子数组」

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
双指针专题(十):恰好等于的困境——「K 个不同整数的子数组」

场景想象:你是一个统计员,要把数组切成很多段。 老板要求:每一段里必须恰好包含K种不同的数字。

  • 比如[1, 2, 1, 2, 3],K=2

  • [1, 2, 1, 2]是符合的(只有 1 和 2 两种)。

  • [1, 2, 1, 2, 3]不符合(有 3 种)。

  • [1]不符合(只有 1 种)。

难点:滑动窗口最擅长处理的是“最多 K 个”(类似《水果成篮》)或者“至少 K 个”。 对于“恰好 K 个”,窗口的左边界很难确定。因为“恰好 2 个”的情况可能有很多种(比如[1, 2][1, 2, 1]都是),左指针到底缩到哪里才算完呢?

力扣 992. K 个不同整数的子数组

https://leetcode.cn/problems/subarrays-with-k-different-integers/

题目分析:

  • 输入:数组nums,整数k

  • 输出:满足条件的子数组数量。

核心思维:恰好(K) = 最多(K) - 最多(K-1)

这是一个非常经典的集合论思想:

  • 最多 K 种:包含“恰好 1 种”、“恰好 2 种” ... “恰好 K 种”。

  • 最多 K-1 种:包含“恰好 1 种” ... “恰好 K-1 种”。

如果你把这两个集合相减,剩下的不就是“恰好 K 种”了吗?

转化优势:“最多包含 K 种整数的子数组数量”非常简单,就是标准的滑动窗口(和《水果成篮》一模一样)。 我们只需要写一个 helper 函数atMost(k),然后调用两次即可:return atMost(k) - atMost(k - 1);

atMost(k)的计数逻辑:当窗口[left, right]满足“最多 K 种”时,以nums[right]结尾的、满足条件的子数组有多少个? 答案是right - left + 1个。

  • 比如[1, 2](K=2)。以 2 结尾的子数组有[2][1, 2],共 2 个。

  • 这个公式累加起来,就是总数。

代码实现 (JavaScript)

JavaScript

/** * @param {number[]} nums * @param {number} k * @return {number} */ var subarraysWithKDistinct = function(nums, k) { // 核心公式:恰好 K = 最多 K - 最多 K-1 return atMost(nums, k) - atMost(nums, k - 1); }; /** * 辅助函数:求最多包含 k 种不同整数的子数组数量 * 这就是标准的滑动窗口模板(类似水果成篮) */ function atMost(nums, k) { let left = 0; let right = 0; let count = 0; // 记录符合条件的子数组总数 let distinctCount = 0; // 当前窗口有多少种不同的数 // 使用数组代替 Map 统计频率,性能会好很多(题目提示 nums[i] <= 20000) // 如果没有范围限制,可以用 Map const freq = new Array(nums.length + 1).fill(0); while (right < nums.length) { // --- 进窗口 --- if (freq[nums[right]] === 0) { distinctCount++; } freq[nums[right]]++; right++; // --- 出窗口 --- // 如果种类超过 k,必须收缩 while (distinctCount > k) { freq[nums[left]]--; if (freq[nums[left]] === 0) { distinctCount--; } left++; } // --- 核心累加 --- // 此时窗口 [left, right-1] 内的种类 <= k // 那么以 right-1 结尾的子数组数量就是窗口长度 count += right - left; } return count; }

深度模拟

假设nums = [1, 2, 1, 2, 3],k = 2

  1. 计算atMost(2):

    • [1]-> +1

    • [1, 2]-> +2 (子数组:2,1,2)

    • [1, 2, 1]-> +3 (子数组:1,2,1,1,2,1)

    • [1, 2, 1, 2]-> +4

    • 遇到 3 (种类变3) -> 缩左边直到[2, 3]-> +2

    • 总数 A

  2. 计算atMost(1):

    • [1]-> +1

    • 遇到 2 (种类变2) -> 缩左边直到[2]-> +1

    • 遇到 1 (种类变2) -> 缩左边直到[1]-> +1

    • ...

    • 总数 B

  3. 结果A - B就是我们要的答案。

总结

恭喜你!🎉 到这里,我们的双指针与滑动窗口专题就彻底结业了!

我们从最简单的快慢指针(移除元素)开始,一路打怪升级,经过了对撞指针(三数之和)、不定长窗口(最小覆盖子串),最后攻克了单调队列(滑动窗口最大值)和数学转换(K个不同整数)。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/16 18:51:10

CSDN官网下载频道:HunyuanOCR配套工具包资源汇总

HunyuanOCR 配套工具包部署与实践&#xff1a;从模型到落地的全链路解析 在文档数字化浪潮席卷各行各业的今天&#xff0c;如何高效、准确地将纸质材料转化为可编辑、可分析的结构化数据&#xff0c;已成为企业自动化升级的关键一环。传统 OCR 技术虽已广泛应用&#xff0c;但面…

作者头像 李华
网站建设 2026/4/16 17:24:36

LaTeX公式识别新方案:HunyuanOCR + MathJax联动尝试

LaTeX公式识别新方案&#xff1a;HunyuanOCR MathJax联动尝试 在科研、教学和工程实践中&#xff0c;我们经常面对一个令人头疼的问题&#xff1a;如何从一张图片中准确提取出复杂的数学公式&#xff1f;无论是扫描的教材、PPT截图&#xff0c;还是学生手写的作业照片&#xf…

作者头像 李华
网站建设 2026/4/15 20:37:36

网盘直链下载助手去广告版是否安全?不如自建HunyuanOCR服务

网盘直链下载助手去广告版是否安全&#xff1f;不如自建HunyuanOCR服务 在智能设备无处不在的今天&#xff0c;我们每天都在和图片里的文字打交道&#xff1a;从网盘里下载的扫描版合同、手机拍下的发票截图、视频中的字幕片段……如何快速准确地把这些“看得见但无法复制”的信…

作者头像 李华
网站建设 2026/4/9 18:06:53

LaTeX学术写作辅助:用HunyuanOCR提取参考文献信息

LaTeX学术写作辅助&#xff1a;用HunyuanOCR提取参考文献信息 在撰写论文时&#xff0c;你是否曾为一条条手动输入参考文献而感到疲惫&#xff1f;尤其是当面对一页页双栏排版、字体细小、甚至带有模糊扫描痕迹的PDF截图时&#xff0c;复制粘贴都变得困难重重。更别提那些夹杂着…

作者头像 李华
网站建设 2026/4/16 21:47:45

从GitHub镜像网站获取腾讯混元OCR模型的完整流程解析

从GitHub镜像网站获取腾讯混元OCR模型的完整流程解析 在文档自动化处理需求日益增长的今天&#xff0c;企业对高精度、多语言、端到端的文字识别能力提出了更高要求。传统OCR方案常因检测与识别模块割裂、多语言支持不足、输出非结构化等问题&#xff0c;在复杂场景中频频“翻车…

作者头像 李华
网站建设 2026/4/11 4:16:09

Obsidian插件开发设想:本地OCR识别图片内文字

Obsidian 插件开发设想&#xff1a;本地 OCR 识别图片内文字 在知识工作者的日常中&#xff0c;截图、扫描文档和手写笔记几乎是不可避免的信息来源。无论是从论文中截取一段关键论述&#xff0c;还是拍下会议白板上的草图&#xff0c;这些图像承载着大量有价值的内容——但它们…

作者头像 李华