news 2026/6/10 22:09:33

双指针专题(十):恰好等于的困境——「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/6/10 10:55:51

清华镜像站使用教程:一键拉取HunyuanOCR Docker镜像

清华镜像站加速部署 HunyuanOCR&#xff1a;一条命令跑通国产端到端 OCR 在智能文档处理需求激增的今天&#xff0c;企业与开发者对OCR技术的要求早已不止“识别文字”这么简单。面对复杂版式、多语言混排、字段精准抽取等现实挑战&#xff0c;传统级联式OCR方案越来越显得力不…

作者头像 李华
网站建设 2026/6/10 10:56:01

UltraISO引导U盘制作含HunyuanOCR Linux系统的可行性

UltraISO引导U盘制作含HunyuanOCR Linux系统的可行性 在政府档案数字化现场&#xff0c;一名工作人员将U盘插入老旧台式机——这台设备既无管理员权限&#xff0c;也未安装任何AI框架。30秒后&#xff0c;系统自动启动一个轻量Linux环境&#xff0c;浏览器弹出HunyuanOCR的Web界…

作者头像 李华
网站建设 2026/6/10 10:54:11

JavaScript Base64编码图片上传至HunyuanOCR接口

JavaScript Base64编码图片上传至HunyuanOCR接口 在智能办公和文档数字化浪潮席卷各行各业的今天&#xff0c;用户对“拍一下就能识别文字”的体验早已习以为常。无论是扫描合同、翻译外文标签&#xff0c;还是从身份证中提取信息&#xff0c;背后都离不开OCR技术的支持。但如何…

作者头像 李华
网站建设 2026/6/10 9:58:39

GitHub镜像网站推荐列表:稳定获取HunyuanOCR及其他AI模型

GitHub镜像网站推荐&#xff1a;高效获取HunyuanOCR等AI模型的实用指南 在当前AI技术快速落地的大背景下&#xff0c;开发者最常遇到的一个“小问题”却可能成为项目推进的“大瓶颈”——如何稳定、快速地下载托管在GitHub上的大型AI模型&#xff1f;尤其是像腾讯推出的Hunyuan…

作者头像 李华
网站建设 2026/6/10 12:35:13

计算机毕业设计springboot大学生心理健康咨询系统 基于Spring Boot的大学生心理健康咨询平台设计与实现 Spring Boot框架下大学生心理健康咨询管理系统开发

计算机毕业设计springboot大学生心理健康咨询系统jpmyh &#xff08;配套有源码 程序 mysql数据库 论文&#xff09; 本套源码可以在文本联xi,先看具体系统功能演示视频领取&#xff0c;可分享源码参考。随着互联网技术的飞速发展&#xff0c;大学生的心理健康问题逐渐受到广泛…

作者头像 李华