news 2026/6/23 8:02:40

算法基础-(单调队列)

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
算法基础-(单调队列)

单调队列

1.什么是单调队列?

单调队列,顾名思义,就是存储的元素要么单调递增要么单调递减的队列。注意,这⾥的队列和普通 的队列不⼀样,是⼀个双端队列。

2.单调队列解决的问题

⼀般⽤于解决滑动窗⼝内最⼤值最⼩值问题,以及优化动态规划。

2.1【模板】单调队列

题⽬来源: 洛⾕
题⽬链接:P1886 滑动窗⼝ /【模板】单调队列
难度系数: ★★

题目描述

有一个长为 n 的序列 a,以及一个大小为 k 的窗口。现在这个窗口从左边开始向右滑动,每次滑动一个单位,求出每次滑动后窗口中的最小值和最大值。

例如,对于序列 [1,3,−1,−3,5,3,6,7] 以及 k=3,有如下过程:

窗口位置[1 3 -1] -3 5 3 6 7 1 [3 -1 -3] 5 3 6 7 1 3 [-1 -3 5] 3 6 7 1 3 -1 [-3 5 3] 6 7 1 3 -1 -3 [5 3 6] 7 1 3 -1 -3 5 [3 6 7]​最小值−1−3−3−333​最大值335567​​

输入格式

输入一共有两行,第一行有两个正整数 n,k;
第二行有 n 个整数,表示序列 a。

输出格式

输出共两行,第一行为每次窗口滑动的最小值;
第二行为每次窗口滑动的最大值。

输入输出样例

输入 #1复制

8 3 1 3 -1 -3 5 3 6 7

输出 #1复制

-1 -3 -3 -3 3 3 3 3 5 5 6 7

说明/提示

【数据范围】
对于 50% 的数据,1≤n≤105;
对于 100% 的数据,1≤k≤n≤106,ai​∈[−231,231)。


【解法】

窗⼝内最⼤值:
从左往右遍历元素,维护⼀个单调递减的队列:
当前元素进队之后,注意维护队列内的元素在⼤⼩为 k 的窗⼝内;
此时队头元素就是最⼤值。
窗⼝内最⼩值:
从左往右遍历元素,维护⼀个单调递增的队列:
当前元素进队之后,注意维护队列内的元素在⼤⼩为 k 的窗⼝内;
此时队头元素就是最⼩值。

【参考代码】

#include <iostream> #include <deque> // 行2:导入双端队列(可以从队头/队尾删元素) using namespace std; const int N = 1e6 + 10; // 行4:序列最长100万+10个数字 int n, k; // 行5:n是序列长度,k是窗口大小 int a[N]; // 行6:存储序列的数字(a[1]是第一个,a[2]第二个...) int main() // 行7:程序入口 { cin >> n >> k; // 行9:读入n和k(比如8和3) for(int i = 1; i <= n; i++) cin >> a[i]; // 行10:读入序列,存到a[1]-a[8] deque<int> q; // 行12:双端队列,存的是数字的“下标”(不是数字本身) // 第一部分:算窗口最小值(维护单调递增队列) for(int i = 1; i <= n; i++) // 行15:遍历每个数字(从第1个到第8个) { // 行17:如果队列不为空,且队尾下标对应的数字 >= 当前数字 → 删掉队尾 while(q.size() && a[q.back()] >= a[i]) q.pop_back(); q.push_back(i); // 行18:把当前数字的下标加入队尾 // 行20:判断队列里的数字是否超出窗口范围(窗口长度不能超过k) if(q.back() - q.front() + 1 > k) q.pop_front(); // 行22:只有当i >= k时(窗口已经滑够3个数字),才输出队头的最小值 if(i >= k) cout << a[q.front()] << " "; } cout << endl; // 行24:最小值输出完,换行 // 第二部分:算窗口最大值(维护单调递减队列) q.clear(); // 行27:清空队列,重新用 for(int i = 1; i <= n; i++) // 行28:再次遍历每个数字 { // 行30:如果队列不为空,且队尾下标对应的数字 <= 当前数字 → 删掉队尾 while(q.size() && a[q.back()] <= a[i]) q.pop_back(); q.push_back(i); // 行31:把当前数字的下标加入队尾 // 行33:判断是否超出窗口范围,超出则删队头 if(q.back() - q.front() + 1 > k) q.pop_front(); // 行35:i >= k时,输出队头的最大值 if(i >= k) cout << a[q.front()] << " "; } cout << endl; // 行37:最大值输出完,换行 return 0; // 行38:程序结束 }

2.2质量检测

题⽬来源: 洛⾕
题⽬链接:P2251 质量检测
难度系数:★★

题目描述

为了检测生产流水线上总共 N 件产品的质量,我们首先给每一件产品打一个分数 A 表示其品质,然后统计前 M 件产品中质量最差的产品的分值 Qm​=min{A1​,A2​,⋯,Am​},以及第 2 至第 M+1 件的 Qm+1​,Qm+2​…… 最后统计第 N−M+1 至第 N 件的 Qn​。根据 Q 再做进一步评估。

请你尽快求出 Q 序列。

输入格式

输入共两行。

第一行共两个数 N、M,由空格隔开。含义如前述。

第二行共 N 个数,表示 N 件产品的质量。

输出格式

输出共 N−M+1 行。

第 1 至 N−M+1 行每行一个数,第 i 行的数 Qi+M−1​。含义如前述。

输入输出样例

输入 #1复制

10 4 16 5 6 9 5 13 14 20 8 12

输出 #1复制

5 5 5 5 5 8 8

说明/提示

[数据范围]

对于 30% 的数据,N≤1000。

对于 100% 的数据,M≤N≤105,Ai​≤106。


【解法】

滑动窗⼝内的最⼩值~

#include <iostream> #include <deque> using namespace std; const int N = 1e6 + 10; // 最多支持100万+10个产品 int n, k; // n是产品总数,k是窗口大小(对应题目里的M) int a[N]; // 存储每个产品的质量分(a[1]是第1个,a[2]是第2个...) int main() { cin >> n >> k; // 读入n=10,k=4 deque<int> q; // 双端队列,存的是“产品的下标”(不是分数本身) // 遍历每个产品(从第1个到第10个) for(int i = 1; i <= n; i++) { cin >> a[i]; // 读入当前产品的分数(比如i=1时读16,i=2时读5...) // 核心1:维护队列“从小到大”(单调递增),保证队头是最小值 // 如果队列不为空,且队尾下标对应的分数 ≥ 当前分数 → 删掉队尾 while(q.size() && a[q.back()] >= a[i]) q.pop_back(); q.push_back(i); // 把当前产品的下标加入队尾 // 核心2:保证队列里的下标都在窗口内(窗口长度不能超过k) // 当前下标 - 队头下标 +1 > k → 队头超出窗口,删掉 if(i - q.front() + 1 > k) q.pop_front(); // 核心3:窗口滑够k个产品后,输出队头对应的最小值(每行一个) if(i >= k) cout << a[q.front()] << endl; } return 0; }

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

解决‘此扩展程序不再受支持’问题:FLUX.1-dev开发环境兼容性优化方案

FLUX.1-dev开发环境兼容性优化&#xff1a;从问题到实践的深度解析 在浏览器插件开发的世界里&#xff0c;一个看似无害的提示——“此扩展程序不再受支持”——往往能让整个项目陷入停滞。尤其是当它出现在你基于最新AI模型构建的文生图工具中时&#xff0c;那种挫败感尤为强烈…

作者头像 李华
网站建设 2026/6/21 21:23:03

火山引擎AI大模型生态中FLUX.1-dev的独特定位分析

火山引擎AI大模型生态中FLUX.1-dev的独特定位分析 在AIGC浪潮席卷内容创作领域的今天&#xff0c;一个核心问题始终困扰着从业者&#xff1a;如何让AI真正“听懂”复杂的视觉指令&#xff1f;无论是广告设计师反复修改提示词却得不到理想构图&#xff0c;还是电商平台需要批量生…

作者头像 李华
网站建设 2026/6/22 23:26:36

抖音直播回放永久保存指南:告别内容丢失的烦恼

抖音直播回放永久保存指南&#xff1a;告别内容丢失的烦恼 【免费下载链接】douyin-downloader 项目地址: https://gitcode.com/GitHub_Trending/do/douyin-downloader 还在为错过精彩直播而懊恼吗&#xff1f;&#x1f914; 当你看到心仪主播的直播&#xff0c;想要永…

作者头像 李华
网站建设 2026/6/22 18:45:48

Bypass Paywalls Clean完整使用教程:快速解锁全网付费内容

Bypass Paywalls Clean是一款专为Chrome浏览器设计的强大扩展工具&#xff0c;能够智能绕过各类网站的付费墙限制&#xff0c;让您免费访问原本需要付费订阅的优质内容。无论您是新闻阅读者、学术研究者还是商业分析师&#xff0c;这款工具都能为您提供便捷的内容获取体验。 【…

作者头像 李华
网站建设 2026/6/23 5:08:47

国产CAD实现铸造与热处理工艺的标准化控制

铸造、热处理等特种工艺&#xff0c;其质量在很大程度上依赖于对过程参数&#xff08;如温度、时间&#xff09;的精确控制。过去&#xff0c;这些参数多依赖于老师傅的个人经验&#xff0c;存在波动性。为实现质量的稳定与均一&#xff0c;必须将个人经验转化为可重复、可验证…

作者头像 李华
网站建设 2026/6/23 19:18:00

微PE官网同款推荐!HunyuanVideo-Foley模型运行环境快速搭建工具包

微PE官网同款推荐&#xff01;HunyuanVideo-Foley模型运行环境快速搭建工具包 在短视频日活突破十亿、影视工业化加速推进的今天&#xff0c;一个被长期忽视却至关重要的环节正成为内容生产链上的“隐形瓶颈”——音效设计。你有没有遇到过这样的场景&#xff1a;精心剪辑了五分…

作者头像 李华