博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
左神进阶2窗口
阅读量:3880 次
发布时间:2019-05-23

本文共 2433 字,大约阅读时间需要 8 分钟。

左神进阶窗口

输入图片说明

窗口内最大值或者最小值的更新结构(单调双向队列):

窗口概念本身并不难理解,窗口从左向右滑动,窗口本身相当于一个固定长度的窗子,依次向右边滑动,从而在中间会有左边减值,右边增加值。

l不能左走,r不能左走。l<r;

窗口内最大值的求解

双端队列,代价都是O(1),不需要这个结构表示第几个数是最大值,只需要双端队列可以依次增加或减少数值。 双端队列中每次保存两个值,当前的数值和当前的位置信息。 加数的逻辑:

加入的数>=双端队列最小(右)值,则从最小值弹出直到 {加入的数>双端队列的数}

每次小的值从右边进入双端队列,整个队列形成从大到小排列的顺序。

如果出现新的数值大于最右边的小数值,则将小的数值弹出,直到放得下新的数值。

如果出现相等的数值,则将先前的数值弹出,将新的数值保存 减数的逻辑: L移动的时候,分析当前最大值所在的位置信息是否过期,如果过期则弹出,没过期则显示当前的最大值。

加数的逻辑中的原理解释:每次增加新的值的时候,如果新增的值比当前双端队列中保存的值小,则需要保留,因为窗口滑动过去之前的较大值之后后面现在的较小的值还有可能变成最大值。当新来了一个较大的值,比队列中的一些值大或相等,则可以直接将那些值删掉,因为接下来那些值存在的时候,新来的值一定存在,并且比那些值大,所以可以直接删掉。 对于双端队列,可以只保存下角标,然后从原始数组中去读取数据值。

public static int[] getMaxWindow(int[] arr,int w){    if(arr == null || w<1 || arr.length 
qmax = new LinkedList
(); int[] res = new int[arr.length-w+1]; int index = 0; for(int i = 0;i
=w-1){ res[index++] = arr[qmax.peekFirst()]; } } return res;}

最大值减去最小值小于或等于num的子数组数量 给定数组arr和整数num,返回一段数组中最大值减去最小值小于num的数目,要求:数组长度为N,请实现时间复杂度为O(N)的解法。

所有子数组的数目等于等差数列n(n-1)/2。先给出一种暴力解法,暴力解法中,其实就是首先两个for循环遍历所有的子数组,然后在所有的过程中进行一个最大减最小。复杂度O(N^3),很高,可能不得分。

public static int getNum1(int[] arr,int num){    int res = 0;    for(int start = 0;start 

新解法:如果中间某个数组从L——R达标,则任何其中的子数组也全部达标。同理,如果一个子数组不达标,则将其任意向外扩,则那些数组也不达标,利用这样的性质整双端队列。

L停留在0位置,如果一直达标,则将R向右扩,直到扩到某个位置X,再往后扩一个,则不达标。 窗口内最大值的更新结构和窗口内的最小值更新结构可以很简单获得最大和最小值,从而进行计算。 以0开头的子数组达标数量则计算了出来,一共有x+1长度的数组达标,他的所有子数组全部达标。 将L来到1位置,此时将窗口最大值和最小值更新,此时R可能仍然可以向外扩,此时右可以得到以1位置开头的所有可行的数组 因为没有回退,一直是当前位置向后走,总的复杂度为O(N)。 新解法

public static int getNum(int[] arr,int num){    if(arr == null || arr.length == 0){        return 0;    }    LinkedList
qmin = new LinkedList
(); LinkedList
qmax = new LinkedList
(); int i = 0;//i为start,j为end int j = 0; int res = 0; //整个while循环是将R向右不断扩展 while(i
= arr[j]){ qmin.polllast(); } qmin.addLast(j); while(!qmax.isEmpty() && arr[qmax.peekLast()] <= arr[j]){ qmax.pollLast(); } qmax.addLast(); if(arr[qmax.getFirst()]- arr[qmin.getFirst()] > num){ break; } j++; } //判断是否进行下一个数开头 if(qmin.peekFirst() == i){ qmin.pollFirst; } if(qmax.peekFirst() == i){ qmax.pollFirst; } //一次性获取完以i开头的所有满足的数组 res += j-i; i++; } return res;

 

 

转载地址:http://izzhn.baihongyu.com/

你可能感兴趣的文章
智能硬件开发如何选择低功耗MCU?
查看>>
阿里感悟(十)如何写好简历
查看>>
阿里感悟(十一)如何准备面试
查看>>
软件架构入门
查看>>
80 多个 Linux 系统管理员必备的监控工具
查看>>
OOD的原则
查看>>
Tool to trace local function calls in Linux
查看>>
Linux 下查询 DNS 服务器信息
查看>>
ulimit 里的 file size 的 block 单位是多少?
查看>>
linux下查看端口对应的进程
查看>>
将 gdb 用作函数跟踪器 (Function Tracer)
查看>>
原 GCC一些有用的技巧
查看>>
yum 变量追加的方法
查看>>
2倍速的下一代Bluetooth,「Bluetooth 5」发布
查看>>
Top 10 “Yum” installables to be productive as a developer on Red Hat Enterprise Linux
查看>>
[小技巧] Vim 如果去除 “existing swap file” 警告
查看>>
如何在linux下检测内存泄漏
查看>>
十年生聚,Vim 8.0 发布了!
查看>>
【演歌】加賀の女 歌词翻译
查看>>
東京音頭 (东京音头) 歌词翻译
查看>>