生成窗口最大数组

题目

  有一个整型数组arr和一个大小为w的窗口从数组的最左边滑到最右边,窗口每次向右边滑动一个位置

  例如,数组为[4,3,5,4,3,3,6,7],窗口大小为3时:
  [4 3 5] 4 3 3 6 7      窗口中最大值的伪5
  4 [3 5 4] 3 3 6 7      窗口中最大值的伪5
  4 3 [5 4 3] 3 6 7      窗口中最大值的伪5
  4 3 5 [4 3 3] 6 7      窗口中最大值的伪4
  4 3 5 4 [3 3 6] 7      窗口中最大值的伪6
  4 3 5 4 3 [3 6 7]      窗口中最大值的伪7

要求

如果数组长度为n,窗口的大小为w,则一共产生n-w+1个窗口状态小的最大值

● 输入:整型数组arr,窗口的大小为w
● 输出:一个长度为n-w+1的数组res,res[i]表示每一种窗口状态下的最大值

以本题为例,结果应该返回[5,5,5,4,6,7]

思路

  本题的关键在于利用双端队列来实现窗口的最大值更新,首先生成双端队列qmax,qmax中存数组arr的下标。
假设遍历到arr[i],qmax的放入规则为:
1、如果qmax为空,直接把下标为i的放进qmax,放入过程结束。
2、如果qmax不为空,取出队尾存放的下标,假设为j。
  ①、如果arr[j]>arr[i],直接把下标为i放进qmax,放入过程结束。
  ②、如果arr[j]<=arr[i],把j从qmax中弹出,重复qmax的放入规则。
  也就是说,如果qmax为空,就直接放入当前位置;如果qmax不为空,qmax队尾的位置所代表的值如果比当前值大,将一直弹出队尾的位置,直到qmax的位置所代表的的值比当前值大的,当前位置才放入qmax的队尾。

假设遍历到arr[i],qmax的弹出规则为:
  如果当前qmax的队头的下标等于i-w,说明当前qmax队头的下标已经过期,弹出当前队头的下标即可。
  根据如上的放入和弹出规则,qmax便成了一个维护窗口w的子数组的最大值,更新的结构,下面举例说明一下题目给出的例子。
  1、开始时qmax为空,qmax={}
  2、遍历到arr[0] == 4,将下标0放入qmax,qmax=[0]
  3、遍历到arr[1] == 3,当前qmax的队尾下标为0,又有arr[0]>arr[1],所以将下标1放入qmax的尾部,qmax={0,1}
  4.遍历到arr[2] == 5当前qmax的队尾下标为0,又有arr[0]>arr[1],所以将下标1从qmax的尾部弹出,qmax变为{},将下标2放入qmax,qmax={2}。此时已经遍历到下标2的位置,窗口arr[0..2]出现,当前qmax的队头的下标为2,所以出窗口arr[0..2]的最大是arr[2] (即5)
  5、遍历到arr[3] == 4,当前队尾的下标为2,又有arr[3]>arr[4],所以将下标2放入qmax尾部,qmax={2,3}。窗口arr[1..3]出现,当前qmax队头的下标为2,这个下标没有过期,所以窗口arr[1..3]的最大值是arr[2] (即5)
  6、遍历到arr[4] == 3当前队尾的下标为3,又有arr[3]>arr[4],所以将下标2放入qmax尾部,qmax={2,3,4}。窗口arr[2..4]出现,当前qmax队头的下标为2,这个下标没有过期,所以窗口arr[2..4]的最大值是arr[2] (即5)
  7、遍历到arr[5] == 3,当前qmax的队尾下标为4,又有arr[4] <= arr[5],所以将下标4从qmax尾部弹出,qmax变为{2,3}。当前qmax的队尾下标为3,又有arr[3] > arr[5],所以将5这个下标放入qmax队尾,qmax的变为qmax={2,3,5}, 窗口arr[3..5],当前qmax的队头的下标为2,这个下标已经过期,所以qmax的头部弹出,qmax变为{3,5}。当前qmax队头的下标为3,这个下标没有过期,所以窗口arr[3..5]的最大值为arr[3] (即4)
  8、遍历到arr[6] == 6,当前qmax的队尾下标为5,又有arr[5] <= arr[6],所以将下标5从qmax尾部弹出,qmax变为{3}。当前qmax的队尾下标为3,又有arr[3] <= arr[6],所以将下标3从qmax尾部弹出,qmax变为{}。所以将6这个下标放入qmax队尾,qmax的变为qmax={6}, 窗口arr[4..6],当前qmax的队头的下标为6,这个没有过期,所以窗口arr[4..6]的最大值为arr[6] (即6)
  9、遍历到arr[7] == 7,当前qmax的队尾下标为6,又有arr[6] <= arr[7],所以将下标6从qmax尾部弹出,qmax变为{}。所以将7这个下标放入qmax队尾,qmax的变为qmax={7}, 窗口arr[5..7],当前qmax的队头的下标为7,这个没有过期,所以窗口arr[5..7]的最大值为arr[7] (即7)
  10、依次出现的窗口最大值为[5,5,5,4,6,7],在遍历过程中收集起来,最后返回即可。

代码演示

package com.itz.zcy.stackAndQueue;

import java.util.Arrays;
import java.util.LinkedList;

/**
 * 有一个整型数组arr和一个大小为w的窗口从数组的最左边滑到最右边,窗口每次向右边滑动一个位置
 * 例如,数组为[4,3,5,4,3,3,6,7],窗口大小为3时:
 *  [4 3 5] 4 3 3 6 7           窗口中最大值的伪5
 *  4 [3 5 4] 3 3 6 7           窗口中最大值的伪5
 *  4 3 [5 4 3] 3 6 7           窗口中最大值的伪5
 *  4 3 5 [4 3 3] 6 7           窗口中最大值的伪4
 *  4 3 5 4 [3 3 6] 7           窗口中最大值的伪6
 *  4 3 5 4 3 [3 6 7]           窗口中最大值的伪7
 *
 *如果数组长度为n,窗口的大小为w,则一共产生n-w+1个窗口状态小的最大值
 *  输入:整型数组arr,窗口的大小为w
 *  输出:一个长度为n-w+1的数组res,res[i]表示每一种窗口状态下的最大值
 *  结果应该返回[5,5,5,4,6,7]
 */
public class MaxWindows {

    /**
     * 使用LinkedList集合来做双端队列,对队头和队尾元素进行弹出,或者加入队尾
     * @param arr 要求的数组
     * @param w 窗口大小
     * @return
     */
    public static int[] getMaxWindows(int[] arr, int w) {
//        异常判定 进行判空操作
        if (arr == null || arr.length < w) {
            return null;
        }
//        创建一个双端队列,已就是linkedList集合
        LinkedList<Integer> qmax = new LinkedList<>();
//        创建一个用来存最大值的数组
        int[] res = new int[arr.length - w + 1];
        int index = 0;
        for (int i = 0; i < arr.length; i++) {
//            当队尾下标,所对应的值小于当前的arr[i] 弹出
            while (!qmax.isEmpty() && arr[qmax.peekLast()] <= arr[i]) {
                qmax.pollLast();
            }
//            进入队列
            qmax.addLast(i);
//            判定当前的下标是否过期
            if (qmax.peekFirst() == i - w) {
                qmax.pollFirst();
            }
//            从arr数组的w-1个数组res才开始存数,此时窗口才满
            if (i >= w - 1) {
                res[index++] = arr[qmax.peekFirst()];
            }
        }
        return res;
    }

    public static void main(String[] args) {
        int[] arr = {4,3,5,4,3,3,6,7};
        int w = 3;
        int[] maxWindows = getMaxWindows(arr, w);
        System.out.println(Arrays.toString(maxWindows)); //[5, 5, 5, 4, 6, 7]

    }
}

总结

假设数组的长度为N,窗口的大小为w,一般情况我们写的都是O(N x w)的时间复杂度,但是采用看双端队列后,每一个下标最多对qmax进一次,出qamx一次。所以遍历过程双端队列的时间复杂度为O(N),整体的时间复杂度也为O(N)

文献:左程云著 《程序员代码面试指南IT名企算法与数据结构题目最优解》(第二版)
版权声明:此文版权归作者所有!

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,001评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,210评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,874评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,001评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,022评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,005评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,929评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,742评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,193评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,427评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,583评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,305评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,911评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,564评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,731评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,581评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,478评论 2 352

推荐阅读更多精彩内容

  • 题目 有一个整形数组arr和一个大小为w的窗口从数组的最左边滑到最右边,窗口每次向右边滑一个位置。  例如:数组为...
    永志阅读 160评论 0 0
  • 题目 有一个整型数组arr和一个大小为w的窗口从数组的最左边滑到最右边,窗口每次向右边滑一个位置。 例如,数组为[...
    50fc16abfd49阅读 1,053评论 0 1
  • 【题目】 有一个整型数组arr和一个大小为w的窗口从数组的最左边滑到最右边,窗口每次向右边滑一个位置。 例如,数组...
    某非著名程序员阅读 208评论 0 1
  • 题目 有一个整型数组arr和一个大小为w的窗口从数组的最左边滑到最右边,窗口每次向右边滑一个位置。 例如,数组为【...
    囧略囧阅读 866评论 1 0
  • 题目 有一个整型数组 arr 和一个大小为 w 的窗口从数组的最左边滑到最右边,窗口每次向右边滑一个位置。 返回一...
    IT_Matters阅读 2,446评论 0 3