【算法】单调队列 - 基础与应用-滑动窗口最大值

题目

给定一个数组 nums,有一个大小为 k 的滑动窗口从数组的最左侧移动到数组的最右侧。你只可以看到在滑动窗口内的 k 个数字。滑动窗口每次只向右移动一位。

返回滑动窗口中的最大值。

思路

暴力:遍历一遍的过程中每次从窗口找到最大的数组,O(n * k)

需要一个队列,在这个队列中放进窗口里的元素,然后随着窗口的移动,队列也一进一出,每次移动后,队列告诉我们里面的最大值是什么。

单调队列,即单调递减或单调递增的队列

保存队列中的元素单调递增或者单调递减,需要自己去实现这个单调队列。

java中使用 Deque deque = new LinkedList<>(); 实现队列

使用Deque自定义一个单调队列插入,弹出,获取一个元素。

  1. 将前k个元素放入单调队列中,取出第一个最大元素
  2. 遍历接下来的数组元素,尝试将窗口中即将被移除的元素从单调队列弹出
  3. 尝试将进入窗口中的元素加入到单调队列
  4. 取出单调队列的最大值,即第一个元素,保存到数组中

始终维护了一个单调的队列,从而每次很容易得取出当前窗口中得最大值。

java 复制代码
class Solution {
    public int[] maxSlidingWindow(int[] nums, int k) {
     
     int n = nums.length;
     int j = 0;
     //存放结果
     int[] res = new int[n - k + 1];
     
    CustomQueue queue = new CustomQueue();

     //先将数组前k的元素放入自定义单调队列中
     for(int i = 0 ; i < k ; i++){
        queue.add(nums[i]);
     }
    
     res[j++] = queue.getMaxValue();

     //遍历数组   i始终是位于滑动窗口最右边
     for(int i = k ; i < n ; i++){
        
         //1.滑动窗口中被移除的元素在单调队列尝试弹出
         queue.poll(nums[i-k]);
         //2.滑动窗口中新加入的元素在单调队列中尝试加入
         queue.add(nums[i]);
         //3.记录当前滑动窗口最大值
         res[j++] = queue.getMaxValue();
     
     }  

     return res;
    }
}


//自定义单调队列
class CustomQueue{
   
  Deque<Integer> deque = new LinkedList<>();

   //弹出元素,如果出口元素和滑动窗口的值一样则弹出
   public void poll(int val){
    if(!deque.isEmpty() && val == deque.peekFirst()){
        deque.pollFirst();
    }
   }
   
   //放入元素.如果入口元素的值小于滑动窗口的值,则弹出入口元素,直到满足为止
   public void add(int val){
    while(!deque.isEmpty() && deque.peekLast() < val){
        deque.pollLast();
    }

    deque.addLast(val); 
   }

   //获取单调队列中最大值
   public int getMaxValue(){
    return deque.peekFirst();
   }

}
相关推荐
hans汉斯9 分钟前
【人工智能与机器人研究】基于力传感器坐标系预标定的重力补偿算法
人工智能·算法·机器人·信号处理·深度神经网络
邓不利东43 分钟前
Spring中过滤器和拦截器的区别及具体实现
java·后端·spring
草履虫建模1 小时前
Redis:高性能内存数据库与缓存利器
java·数据库·spring boot·redis·分布式·mysql·缓存
苹果醋31 小时前
Vue3组合式API应用:状态共享与逻辑复用最佳实践
java·运维·spring boot·mysql·nginx
vortex52 小时前
算法设计与分析:分治、动态规划与贪心算法的异同与选择
算法·贪心算法·动态规划
Micro麦可乐2 小时前
Java常用加密算法详解与实战代码 - 附可直接运行的测试示例
java·开发语言·加密算法·aes加解密·rsa加解密·hash算法
掉鱼的猫2 小时前
Java MCP 鉴权设计与实现指南
java·openai·mcp
前端拿破轮2 小时前
🤡🤡🤡面试官:就你这还每天刷leetcode?连四数相加和四数之和都分不清!
算法·leetcode·面试
努力的小郑2 小时前
Spring三级缓存硬核解密:二级缓存行不行?一级缓存差在哪?
java·spring·面试
手握风云-2 小时前
JavaEE初阶第七期:解锁多线程,从 “单车道” 到 “高速公路” 的编程升级(五)
java·开发语言