news 2026/1/29 13:44:09

双指针专题(十):恰好等于的困境——「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/1/22 16:49:02

CSDN官网问答区热门:HunyuanOCR怎么读取旋转文本?

HunyuanOCR如何高效识别旋转文本&#xff1f;技术解析与实践指南 在智能文档处理日益普及的今天&#xff0c;一个看似简单却极具挑战的问题正困扰着无数开发者&#xff1a;手机随手一拍的发票、倾斜扫描的合同、视频中斜向滚动的字幕——这些旋转文本该如何被准确读取&#xff…

作者头像 李华
网站建设 2026/1/23 6:55:43

基于腾讯混元OCR的网页推理部署指南(支持4090D单卡)

基于腾讯混元OCR的网页推理部署指南&#xff08;支持4090D单卡&#xff09; 在企业数字化转型加速的今天&#xff0c;如何高效、低成本地处理海量图像中的文字信息&#xff0c;已成为一个普遍痛点。传统的OCR方案往往依赖多阶段流水线——先检测文本区域&#xff0c;再识别内容…

作者头像 李华
网站建设 2026/1/24 18:43:46

网盘直链下载助手原理剖析及其与HunyuanOCR的结合可能

网盘直链下载助手原理剖析及其与HunyuanOCR的结合可能 在数字办公和在线学习日益普及的今天&#xff0c;我们每天都在处理大量来自微信、邮件或群聊中的截图——一张张包含网盘链接的图片&#xff0c;背后是亟待获取的教学资料、项目文档或共享资源。手动复制链接、输入提取码、…

作者头像 李华
网站建设 2026/1/28 6:47:00

LaTeX排版自动化:HunyuanOCR识别手写公式转LaTeX代码实验

LaTeX排版自动化&#xff1a;HunyuanOCR识别手写公式转LaTeX代码实验 在高校的自习室里&#xff0c;一个学生正埋头推导微分方程&#xff0c;草稿纸上密密麻麻写满了符号和积分表达式。他想把这页内容整理进论文&#xff0c;却不得不一边翻LaTeX手册&#xff0c;一边敲击键盘输…

作者头像 李华
网站建设 2026/1/29 11:39:37

Faststone Capture替代方案:基于HunyuanOCR的截图识别工具开发

Faststone Capture替代方案&#xff1a;基于HunyuanOCR的截图识别工具开发 在每天处理大量文档、会议截图和跨语言资料的办公场景中&#xff0c;你是否也曾遇到这样的困扰&#xff1f;——看到一段关键信息藏在一张模糊的PPT截图里&#xff0c;复制不了&#xff1b;收到一份扫描…

作者头像 李华
网站建设 2026/1/24 17:01:21

UltraISO高级功能探讨:挂载包含HunyuanOCR的ISO镜像

UltraISO高级功能探讨&#xff1a;挂载包含HunyuanOCR的ISO镜像 在当今AI模型部署日益复杂的背景下&#xff0c;如何让一个高性能OCR系统像U盘一样“即插即用”&#xff0c;成为许多企业与开发者关注的核心问题。尤其是在金融票据处理、合同信息提取、多语言文档识别等对准确率…

作者头像 李华