题目描述

给定一个数组和滑动窗口的大小,找出所有滑动窗口里数值的最大值。例如,如果输入数组{2,3,4,2,6,2,5,1}及滑动窗口的大小3,那么一共存在6个滑动窗口,他们的最大值分别为{4,4,6,6,6,5}; 针对数组{2,3,4,2,6,2,5,1}的滑动窗口有以下6个: {[2,3,4],2,6,2,5,1}, {2,[3,4,2],6,2,5,1}, {2,3,[4,2,6],2,5,1}, {2,3,4,[2,6,2],5,1}, {2,3,4,2,[6,2,5],1}, {2,3,4,2,6,[2,5,1]}。

思路

思路一:

用双端队列来存储数组元素的索引

  1. 如果新来的值比队列尾部的数小,那就追加到后面,因为它可能在前面的最大值划出窗口后成为最大值
  2. 如果新来的值比尾部的大,那就删掉尾部,再追加到后面
  3. 如果追加的值比的索引跟队列头部的值的索引超过窗口大小,那就删掉头部的值
  4. 每次队列的头都是滑动窗口中值最大的

思路二:

最大堆方法

构建一个窗口size大小的最大堆,每次从堆中取出窗口的最大值,随着窗口往右滑动,需要将堆中不属于窗口的堆顶元素删除。

代码实现

  1. package StackAndQueue;
  2. import java.util.ArrayList;
  3. import java.util.Deque;
  4. import java.util.LinkedList;
  5. import java.util.PriorityQueue;
  6. /**
  7. * 滑动窗口的最大值
  8. * 给定一个数组和滑动窗口的大小,找出所有滑动窗口里数值的最大值。例如,如果输入数组{2,3,4,2,6,2,5,1}及滑动窗口的大小3,那么一共存在6个滑动窗口,他们的最大值分别为{4,4,6,6,6,5}; 针对数组{2,3,4,2,6,2,5,1}的滑动窗口有以下6个: {[2,3,4],2,6,2,5,1}, {2,[3,4,2],6,2,5,1}, {2,3,[4,2,6],2,5,1}, {2,3,4,[2,6,2],5,1}, {2,3,4,2,[6,2,5],1}, {2,3,4,2,6,[2,5,1]}。
  9. */
  10. public class Solution52 {
  11. public static void main(String[] args) {
  12. Solution52 solution52 = new Solution52();
  13. int[] num = {2, 3, 4, 2, 6, 2, 5, 1};
  14. int size = 3;
  15. ArrayList<Integer> list = solution52.maxInWindows(num, size);
  16. System.out.println(list);
  17. }
  18. /**
  19. * 最大堆方法
  20. * 构建一个窗口size大小的最大堆,每次从堆中取出窗口的最大值,随着窗口往右滑动,需要将堆中不属于窗口的堆顶元素删除。
  21. *
  22. * @param num
  23. * @param size
  24. * @return
  25. */
  26. public ArrayList<Integer> maxInWindows_2(int[] num, int size) {
  27. ArrayList<Integer> res = new ArrayList<>();
  28. if (size > num.length || size < 1) return res;
  29. // 构建最大堆,即堆顶元素是堆的最大值。
  30. PriorityQueue<Integer> heap = new PriorityQueue<Integer>((o1, o2) -> o2 - o1);
  31. for (int i = 0; i < size; i++) heap.add(num[i]);
  32. res.add(heap.peek());
  33. for (int i = 1; i + size - 1 < num.length; i++) {
  34. heap.remove(num[i - 1]);
  35. heap.add(num[i + size - 1]);
  36. res.add(heap.peek());
  37. }
  38. return res;
  39. }
  40. /**
  41. * 双队列方法
  42. * 滑动窗口的最大值总是保存在队列首部,队列里面的数据总是从大到小排列。
  43. *
  44. * @param num
  45. * @param size
  46. * @return
  47. */
  48. public ArrayList<Integer> maxInWindows(int[] num, int size) {
  49. ArrayList<Integer> res = new ArrayList<>();
  50. if (num == null || num.length == 0 || size == 0 || size > num.length) {
  51. return res;
  52. }
  53. Deque<Integer> deque = new LinkedList<>();
  54. for (int i = 0; i < num.length; i++) {
  55. if (!deque.isEmpty()) {
  56. // 如果队列头元素不在滑动窗口中了,就删除头元素
  57. if (i >= deque.peek() + size) {
  58. deque.pop();
  59. }
  60. // 如果当前数字大于队列尾,则删除队列尾,直到当前数字小于等于队列尾,或者队列空
  61. while (!deque.isEmpty() && num[i] >= num[deque.getLast()]) {
  62. deque.removeLast();
  63. }
  64. }
  65. deque.offer(i); // 入队列
  66. // 滑动窗口经过一个滑动窗口的大小,就获取当前的最大值,也就是队列的头元素
  67. if (i + 1 >= size) {
  68. res.add(num[deque.peek()]);
  69. }
  70. }
  71. return res;
  72. }
  73. }

剑指Offer-滑动窗口的最大值的更多相关文章

  1. 剑指offer——滑动窗口的最大值

    给定一个数组和滑动窗口的大小,找出所有滑动窗口里数值的最大值.例如,如果输入数组{2,3,4,2,6,2,5,1}及滑动窗口的大小3,那么一共存在6个滑动窗口,他们的最大值分别为{4,4,6,6,6, ...

  2. 剑指0ffer59.滑动窗口的最大值

    给定一个数组 nums 和滑动窗口的大小 k,请找出所有滑动窗口里的最大值. 示例: 输入: nums = [1,3,-1,-3,5,3,6,7], 和 k = 3输出: [3,3,5,5,6,7] ...

  3. 剑指offer64:滑动窗口的最大值

    题目描述: 给定一个数组和滑动窗口的大小,找出所有滑动窗口里数值的最大值.例如,如果输入数组{2,3,4,2,6,2,5,1}及滑动窗口的大小3,那么一共存在6个滑动窗口,他们的最大值分别为{4,4, ...

  4. 剑指offer--50.滑动窗口的最大值

    时间限制:1秒 空间限制:32768K 热度指数:157641 题目描述 给定一个数组和滑动窗口的大小,找出所有滑动窗口里数值的最大值.例如,如果输入数组{2,3,4,2,6,2,5,1}及滑动窗口的 ...

  5. 【Java】 剑指offer(59-2) 队列的最大值

      本文参考自<剑指offer>一书,代码采用Java语言. 更多:<剑指Offer>Java实现合集   题目 请定义一个队列并实现函数max得到队列里的最大值,要求函数ma ...

  6. [剑指Offer]59-队列的最大值(题目二待补)

    题目一:滑动窗口的最大值 题目链接 https://www.nowcoder.com/practice/1624bc35a45c42c0bc17d17fa0cba788?tpId=13&tqI ...

  7. 剑指offer——68队列的最大值

    题目描述 给定一个数组和滑动窗口的大小,找出所有滑动窗口里数值的最大值.例如,如果输入数组{2,3,4,2,6,2,5,1}及滑动窗口的大小3,那么一共存在6个滑动窗口,他们的最大值分别为{4,4,6 ...

  8. 剑指offer——69队列的最大值

    题目: 队列的最大值.请定义一个队列并实现函数max得到队列里的最大值,要求函数max.push_back和pop_front的时间复杂度都是O(1). 题解: 使用队列,操持队列的排序为从大到小的顺 ...

  9. 剑指offer 59-II 队列的最大值

    题目描述 请定义一个队列并实现函数 max_value 得到队列里的最大值,要求函数max_value.push_back 和 pop_front 的均摊时间复杂度都是O(1). 若队列为空,pop_ ...

随机推荐

  1. 第三十二篇-NavigationView导航抽屉的使用

    效果图: 导航抽屉所用到的布局是DrawerLayout,可以在里面添加一个线性布局和TextView组件,TextView组件的文本信息就是"主页面".然后和线性布局平行添加一个 ...

  2. springmvc拦截器说明

    一般 我们在spring mvc的配置文件中 这样配置拦截器 <!--拦截器 --> <mvc:interceptors> <!--多个拦截器,顺序执行 --> & ...

  3. 在linux下面解压用的zxpf是什么意思,它跟zxvf有啥区别

    在linux下面解压用的zxpf是什么意思,它跟zxvf有啥区别 linux 命令中tar后跟的zxvf是什么意思:.tar.gz是一个压缩包   .tar只是打包而没有压缩 z:表示 tar 包是被 ...

  4. Kafka Offset相关命令总结

    Kafka Offset相关命令总结 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.查询topic的offset的范围 1>.查询某个topic的offset的最小值 [ ...

  5. java 中数据的强制转换 和计算的补码运算

    原码 反码 补码的定义与运算 1原码: 原码是将十进制或者其他进制的数转换为二进制表示(且要根据数据的类型转换) 如:130 (默认是Int类型,则是4个字节) 原码是:00000000 000000 ...

  6. Linux-CentOS 查看(监控)服务器网卡流量

    1.使用系统自带的命令: watch more /proc/net/dev 2. 使用nload命令,需要自行安装 1)# yum install flex byacc libpcap ncurses ...

  7. Understanding Favicon

    Favicon 简介 Favicon : 是favorites icon 的缩写,被称为website icon . page icon. urlicon. 最初定义一个favicon的方法是将一个名 ...

  8. windows环境配置计划任务让weblogic的servers开机启动【原】

    准备脚本 注意如果weblogic在D盘,那么以下cmd中的所有C:都要替换成D: , 因为windows需要切盘符. 启动weblogic管理服务adminServer 的文件: startAdmi ...

  9. Oracle了解(一)

    通常所说的Oracle数据库服务器由一个数据库和至少一个数据库实例组成. 数据库实例是由系统后台进程和分配的内存区域构成 实例你是提供服务的进程,数据库是存放的数据. 数据库是存储数据的文件 数据库实 ...

  10. 在Linux(CentOS 6.6)服务器上安装并配置基于Apache的SVN服务器

    #!/bin/bash # # 在Linux(CentOS 6.6)服务器上安装并配置基于Apache的SVN服务器: # # .安装服务 # .创建svn版本库 # .创建svn用户 # .配置sv ...