【LeetCode栈与队列#05】滑动窗口最大值
滑动窗口最大值
给定一个数组 nums,有一个大小为 k 的滑动窗口从数组的最左侧移动到数组的最右侧。你只可以看到在滑动窗口内的 k 个数字。滑动窗口每次只向右移动一位。
返回滑动窗口中的最大值。
进阶:
你能在线性时间复杂度内解决此题吗?
提示:
- 1 <= nums.length <= 10^5
- -10^4 <= nums[i] <= 10^4
- 1 <= k <= nums.length
思路
虽然本题在LeetCode上为困难级别,但是似乎很容易想到对应的暴力解法,首先我们得明白这题难在哪里
使用暴力解法
本题的难点在于使用一般的方法解,时间复杂度会很高
一种很自然的想法是:滑动窗口遍历过程中,我们又去遍历窗口内的数,比较大小后返回最大值
那么这种方法的时间复杂度为O(n*k),k为窗口大小,n为数组遍历次数
使用队列行不行?
观察滑动窗口移动的方式,和队列有点像
使用队列模拟滑动窗口,窗口每次向右移动一个单位,只需将队列的首位pop并将下一个数添加到队尾即可完成操作
那么可不可以用优先级队列来获取窗口内的最大值呢?
可以,但是会破坏窗口内元素的顺序,不利于后续操作
例如,当前窗口内的数为{1,3,-1}
如果将其放入优先级队列,那么顺序就会变为{3,1,-1}
很好,找到最大值3了。但是窗口向后移动时,按原来的设想,我们需要把1弹出,但是现在1被夹在中间,pop不了,出现问题。所以不能用优先级队列去做
自定义队列
逻辑
虽然不能用优先级队列去解决本题,但是这个思考过程是可以借鉴的
这里需要明确一点:队列中的元素顺序其实并不重要,队列只是用于模拟滑动窗口的一个数据结构
队列中数据的进出只需要满足滑动窗口移动的规则即可
因此,我们可以设想这样一种自定义队列:
该队列的队头永远是当前滑动窗口中的最大值,当滑动窗口移动时,我们就把新的数添加到队尾。
如果新数比队头的数还要大,那么将之前所有的队内元素弹出,使新数位于队头,如此循环
这种队列被称为单调队列
然后我们再通过一个函数去获取队列首部的最大值就可以实现题目的要求了,最终结果是{3,3,5,5,5,3}
总结一下,我们需要实现的队列结构大致如下:
class MyQueue {
public:
void pop(int value) {
}
void push(int value) {
}
int front() {
return que.front();
}
};
实现
那么想法有了,用什么数据结构去实现这个单调队列呢?--> deque
实现单调队列MyQueue
如上面说的,单调队列类MyQueue中提供pop、push、front三个方法
pop
用于弹出元素
每次弹出之前要先判断当前队列是否为空,以及输入值是否与队头数据相等
void pop(int value){
if(!que.empty() && value == que.front()){
que.pop_front();//弹出队首元素
}
}
如果要pop的元素是队首元素,说明当前最大值需要更新,并且该更新是因为窗口移动导致的,而不是来了个新的最大值
简单来说,不是每次滑动窗口移动都需要调用pop,pop函数只负责单纯的pop掉队列开头的元素
如果有新的数值进来并且是最大值,那么将新的最大值移动到队首的操作会由push函数实现
push
用于将新元素添加到队尾
void push(int value){
while(!que.empty() && value > que.back()){//如果新元素一直大于队尾的数,就不断将队尾数pop
que.pop_back();
}
//直到遇到比新元素更大的值或者只剩下新元素
que.push_back(value);
}
front
用于查询队列头部的元素
// 查询当前队列里的最大值 直接返回队列前端也就是front就可以了。
int front() {
return que.front();
}
完整代码
class MyQueue {
public:
deque<int> que;
void pop(int value) {
if(!que.empty() && value == que.front()){
que.pop_front();//弹出队首元素
}
}
void push(int value) {
while(!que.empty() && value > que.back()){//如果新元素一直大于队尾的数,就不断将队尾数pop
que.pop_back();
}
//直到遇到比新元素更大的值或者只剩下新元素
que.push_back(value);
}
int front() {
return que.front();
}
};
整体代码
步骤:
1、使用deque实现单调队列MyQueue
2、实例化单调队列
3、将整数数组中k(窗口大小)个元素遍历加入队列中
4、res记录当前k个数中的最大值
5、开始移动窗口遍历整数数组
- 使用pop配合push模拟窗口的滑动
- 记录当前最大值
6、返回结果数组res
class Solution {
private:
class MyQueue{
public:
deque<int> que;
void pop(int value){
if(!que.empty() && value == que.front()){
que.pop_front();//弹出队首元素
}
}
void push(int value){
while(!que.empty() && value > que.back()){//如果新元素一直大于队尾的数,就不断将队尾数pop
que.pop_back();
}
//直到遇到比新元素更大的值或者只剩下新元素
que.push_back(value);//注意用的是自己实现的push,不是push_back
}
int front(){// 查询当前队列里的最大值 直接返回队列前端也就是front就可以了。
return que.front();
}
};
public:
vector<int> maxSlidingWindow(vector<int>& nums, int k) {
MyQueue que;
//输出值是数组
vector<int> res;
//将滑动窗口初始位置的值加入
for(int i = 0; i < k; ++i){
que.push(nums[i]);
}//此时执行完后,que中为[3,-1]
//记录当前最大值
res.push_back(que.front());//初始窗口中也要取最大值,不然会漏情况
//模拟窗口移动
for(int i = k; i < nums.size(); ++i){//注意从k开始
que.pop(nums[i - k]);//将窗口最前面的数pop掉
que.push(nums[i]);//往窗口末尾加入新数,使窗口向右移动
res.push_back(que.front());//记录当前窗口的最大值
}
return res;
}
};
主函数部分说明
主函数逻辑:
1、实例化MyQueue、结果数组vector res;
2、初始化滑动窗口,即先使用自定义的push(不是push_back)把数组前k个数放入队列(同时就排好序了)
3、将当前初始化窗口的最大值保存(用vector提供的方法:push_back)
4、模拟窗口移动(注意遍历开始的位置,以及窗口头部和尾部位置)
补充知识
vector添加元素用push_back,deuqe也用这个
TBD
二刷问题
1、自定义的队列类MyQueue记得写作用域public
2、在主代码中维护的que队列不是所谓的"窗口",它只是用来保证我们设想的窗口中最左边的值永远是最大值(队首),可以类比辅助栈的作用
3、一开始我们需要往窗口中填入窗口大小的元素,加入元素的方式是自定义类中的push方法,加完之后也要保存当前队列首部的最大值,不要漏情况
4、遍历是从k开始的,不是0
【LeetCode栈与队列#05】滑动窗口最大值的更多相关文章
- [Leetcode]双项队列解决滑动窗口最大值难题
这道题是从优先队列的难题里面找到的一个题目.可是解法并不是优先队列,而是双项队列deque 其实只要知道思路,这一道题直接写没有太大的问题.我们看看题 给定一个数组 nums,有一个大小为 k 的滑动 ...
- 【Leetcode堆和双端队列】滑动窗口最大值(239)
题目 给定一个数组 nums,有一个大小为 k 的滑动窗口从数组的最左侧移动到数组的最右侧.你只可以看到在滑动窗口内的 k 个数字.滑动窗口每次只向右移动一位. 返回滑动窗口中的最大值. 示例: 输入 ...
- [LeetCode] 239. Sliding Window Maximum 滑动窗口最大值
Given an array nums, there is a sliding window of size k which is moving from the very left of the a ...
- [leetcode]239. Sliding Window Maximum滑动窗口最大值
Given an array nums, there is a sliding window of size k which is moving from the very left of the a ...
- 代码随想录算法训练营day12 | leetcode 239. 滑动窗口最大值 347.前 K 个高频元素
基础知识 ArrayDeque deque = new ArrayDeque(); /* offerFirst(E e) 在数组前面添加元素,并返回是否添加成功 offerLast(E e) 在数组后 ...
- [思维提升|干货All in]6种算法解决LeetCode困难题:滑动窗口最大值
为了更好的阅读体验,欢迎阅读原文: [思维提升|干货All in]6种算法解决LeetCode困难题:滑动窗口最大值 (eriktse.com) 最近在leetcode遇到一道非常经典的题目:239. ...
- Leetcode 239.滑动窗口最大值
滑动窗口最大值 给定一个数组 nums,有一个大小为 k 的滑动窗口从数组的最左侧移动到数组的最右侧.你只可以看到在滑动窗口 k 内的数字.滑动窗口每次只向右移动一位. 返回滑动窗口最大值. 示例: ...
- Java实现 LeetCode 239 滑动窗口最大值
239. 滑动窗口最大值 给定一个数组 nums,有一个大小为 k 的滑动窗口从数组的最左侧移动到数组的最右侧.你只可以看到在滑动窗口内的 k 个数字.滑动窗口每次只向右移动一位. 返回滑动窗口中的最 ...
- 代码随想录第十三天 | 150. 逆波兰表达式求值、239. 滑动窗口最大值、347.前 K 个高频元素
第一题150. 逆波兰表达式求值 根据 逆波兰表示法,求表达式的值. 有效的算符包括 +.-.*./ .每个运算对象可以是整数,也可以是另一个逆波兰表达式. 注意 两个整数之间的除法只保留整数部分. ...
- [Swift]LeetCode239. 滑动窗口最大值 | Sliding Window Maximum
Given an array nums, there is a sliding window of size k which is moving from the very left of the a ...
随机推荐
- Redis 的简单学习与整理
Redis 的简单学习与整理 背景 最近一直进行Redis性能调优和高可用的课题 但是不管什么课题,第一步应该是学习与使用redis 之前总结过 rdb 与 rdr 分析 键值对等内容. 但是发现想要 ...
- docker -- images镜像消失问题排查
1. 问题描叙 安装model-serving组件时,错误日志输出push时对应的tag不存在,导致镜像推送失败 2. 问题排查 # 找到对应镜像,尝试手动推送 docker images|grep ...
- 如何安装typescript-如何运行typescript
如何安装 typescript npm install typescript -g 安装 ts 之前需要安装 node 哈 查看 ts 的版本 tsc -V 我的是 3.6.3 的版本 typescr ...
- 为不同版本python安装pip的正确做法
想学习Python3,但是暂时又离不开Python2.在Windows上如何让它们共存呢? 目前国内网站经常会让大家把其中一个python.exe改个名字(嗯,我也这样讲过,在此纠正一下),这样区分开 ...
- 从零开始配置 vim(6)——缩写
关于vim能快速编辑文本的能力,我们见识到了 operator + motion ,见识到了. 范式和宏.甚至可以使用命令来加快文本编辑.在后面我们又介绍了快捷键绑定来快速执行我们想要的操作.今天我们 ...
- 手撕Vue-数据驱动界面改变上
经过上一篇的介绍,已经实现了监听数据的变化,接下来就是要实现数据变化后,界面也跟着变化,这就是数据驱动界面改变. 想要实现数据变化之后更新UI界面,我们可以使用发布订阅模式来实现,先定义一个观察者类, ...
- MySQL【二】---数据库查询详细教程{查询、排序、聚合函数、分组}
1.数据准备.基本的查询(回顾一下) 创建数据库 create database python_test charset=utf8; 查看数据库: show databases; 使用数据库: use ...
- 【分享】从Mybatis源码中,学习到的10种设计模式
作者:小傅哥 博客:https://bugstack.cn 沉淀.分享.成长,让自己和他人都能有所收获! 一.前言:小镇卷码家 总有不少研发伙伴问小傅哥:"为什么学设计模式.看框架源码.补技 ...
- 深入浅出Java多线程(四):线程状态
引言 大家好,我是你们的老伙计秀才!今天带来的是[深入浅出Java多线程]系列的第四篇内容:线程状态.大家觉得有用请点赞,喜欢请关注!秀才在此谢过大家了!!! 在现代软件开发中,多线程编程已经成为提升 ...
- 在K8S中,节点故障驱逐pod过程时间怎么定义?
在Kubernetes中,节点故障驱逐Pod的过程涉及多个参数和组件的相互作用.以下是该过程的简要概述: 默认设置:在默认配置下,节点故障时,工作负载的调度周期约为6分钟. 关键参数: node-mo ...