【滑动窗口】力扣239.滑动窗口最大值

前面的文章我们练习数十道 动态规划 的题目。相信小伙伴们对于动态规划的题目已经写的 得心应手 了。

还没看过的小伙伴赶快关注一下,学习如何 秒杀动态规划 吧!

接下来我们开启一个新的篇章 —— 「滑动窗口」

滑动窗口

滑动窗口 是一种基于 双指针 思想的算法。两个指针指向的元素之间会形成一个窗口,从前往后遍历元素进行一定的运算。

从名字中也不难看出:滑动 说明窗口的大小并不是固定不变的。通过左右指针按照 规则 向前滑动形成的不同大小的窗口解决具体的问题。

因此,分析问题的 关键 就在于 明确两个指针的移动规则

核心步骤

  1. 初始时,L = R = 0 ,并规定窗口的取值为 [L, R) 即:左闭右开 。因此,初始时[0, 0) 无意义,窗口内没有任何元素被包含。

  2. 循环遍历,在保证不会越界的情况下,不断 增大右指针R。当满足要求后,停止增大右指针R

  3. 左指针L开始 不断增大,直到不满足要求后停下。

  4. 重复执行 2、3 两步,直到右指针R走到尽头( 越界 )。

左右指针轮流向前移动,窗口大小不断变化。新旧元素加入和移出后,及时更新该窗口范围内的相关数据。当执行结束后,收集到了所有符合要求的数据,且时间复杂度降低到了 O ( N ) O(N) O(N)


下面我们通过一道 力扣 Hard 题 进行学习。

239. 滑动窗口最大值

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


在解决该题目之前,我们先来讨论一下这类题的套路。

思路与套路

对于 窗口内最大最小值 的题目,我们采用 双端队列 的结构进行思考。并对 出队入队 进行这样的规定(以窗口内的最大值举例):

队列含义:
  1. 如果此时开始缩窗口(即:L++),哪些值会依次成为此时窗口内的最大值。
  2. 要求,队列中的元素从大到小排列。
队列的出入:

队尾 的出队与入队:

  • 如果队空,直接入队;
  • 否则,如果队尾位置不大于当前来到位置的数,从队尾出队。直到队尾元素大于当前位置元素,从队尾入队。

队头 的出队:

  • 当窗口缩小时,若队头元素已经不在当前的窗口中时,从队头弹出即可。

是不是没搞懂在说什么,没关系。我们通过具体的例子感受该流程:

就拿例 1 的例子,nums=[1,3,-1,-3,5,3,6,7],构造双端队列的值依次为:

注意: 双端队列中 队头元素 保存的是 当前范围最大值

先看从队尾出入队的情况:



以上均是 L 不动,R 在右移的情况。

再看从队头出队的情况:

下面我们再来看 L 也向右移动的情况:

L 指向 1,R 指向 5 时为例:


跟着图细心体会该过程哦!相信小伙伴能够理解双端队列的出入规则了!


下面我们回归正题,解决上面力扣 滑动窗口最大值 的题目。

仔细思考后发现,其实就是将上面的过程 加以限制

限制了窗口的大小必须为 k 。

直接上代码。

代码

public static int[] getMaxWindow(int[] arr, int w) {if (arr == null || w < 1 || arr.length < w) {return null;}// 双端队列LinkedList<Integer> qmax = new LinkedList<Integer>();int N = arr.length;// 最终会产生 N - w + 1 个结果int[] res = new int[N - w + 1];// res 数组的下标int index = 0;// R 从左到右遍历数组,不回退for (int R = 0; R < N; R++) {// 双端队列中有值,并且 队尾的数 比当前的 小 就弹出while (!qmax.isEmpty() && arr[qmax.peekLast()] <= arr[R]) {qmax.pollLast();}// 双端队列为空,或者弹出完比自己小的数了// 从末尾插入队列中qmax.addLast(R);// R-w 是过期的下标,即 L 的位置// 队头的下标 过期了就弹出if (qmax.peekFirst() == R - w) {qmax.pollFirst();}// 能够形成完整的窗口了,开始往 结果数组 里填最终答案if (R >= w - 1) {res[index++] = arr[qmax.peekFirst()];}}return res;
}

代码解释

其实注释已经写的非常清楚了。

  1. 在上面的图中,我么在双端队列中放入的是 元素值,但在实际的代码中,存入的是下标,这样的话既能够比较大小,又能方便的进行入队出队操作。
  2. 窗口大小固定,R 和 L 一起右移。因此若判断出队头元素已经离开了窗口,就要弹出。
  3. 刚开始,R 从 0 开始增大,窗口还未形成。只有当形成 k 大小的窗口后再开始更新答案。(那 思考一下 为什么不直接从下标 k 开始呢?这不直接就有窗口了么?欢迎留言评论 ~)

~ 点赞 ~ 关注 ~ 星标 ~ 不迷路 ~!!!

关注回复「ACM紫书」获取 ACM 算法书籍~

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/525721.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Matlab|配电网智能软开关(sop)规划模型

目录 1 主要内容 目标函数 2 部分程序 3 程序结果 3.1 sop选址定容优化模型 3.2 对比算例&#xff08;不含sop&#xff09; 4 下载链接 1 主要内容 该程序参考文献《基于改进灵敏度分析的有源配电网智能软开关优化配置》&#xff0c;采用二阶锥算法&#xff0c;以改进的…

动态调整html表格每列宽度

为什么想自动计算列宽呢&#xff1f;因为我有一次拿到一个项目&#xff0c;它里面的列宽都是写死的。后来需要改&#xff0c;一个个的改太麻烦了。 诸如这样的表格在网站上非常常见。我们不对列做设置的话&#xff0c;列宽就会取每列文本内容的最大长度。在只有一条文本非常长…

ubuntu安装开源汇编调试器NASM

安装 安装很简单&#xff0c;直接在终端输入以下命令即可 sudo apt-get install nasm 安装完成后&#xff0c;如果可以查看到nasm的版本号即可视为安装成功 nasm -version 测试 创建汇编文件 创建一个asm文件 vim hello.asm 文件内容如下 section .datahello: db …

Linux随记(八)

一、crontab运行shell脚本&#xff0c;py脚本 &#xff08;注意事项&#xff09; 情景描述&#xff1a; 目前有个sh脚本他最初大致内容是。 cat t11.sh#!/bin/bash source /etc/profile /bin/python3 /tmp/1.py sh /tmp/1.sh echo -e "$(date %F)" >…

LeetCode148题:排序链表(python3)

在数组排序中&#xff0c;常见的排序算法有&#xff1a;冒泡排序、选择排序、插入排序、希尔排序、归并排序、快速排序、堆排序、计数排序、桶排序、基数排序等。 而对于链表排序而言&#xff0c;因为链表不支持随机访问&#xff0c;访问链表后面的节点只能依靠 next 指针从头…

2024-03-05 linux 分区老显示满,Use 100%,原因是SquashFS 是一种只读文件系统,它在创建时就已经被填满,所有空间都被使用。

一、这两天一直纠结一个问题&#xff0c;无论怎么修改&#xff0c;linux 分区老显示满&#xff0c;Use 100%&#xff0c;全部沾满。如下图的oem分区。 二、导致出现上面的原因是&#xff1a;SquashFS文件系统里的空间利用率总是显示为100%。 三、SDK里面也说明SquashFS文件系统…

昏暗场景增强-低照度增强-弱光增强(附代码)

引言 随着现代科技的发展&#xff0c;图像采集设备已经渗透到生活的方方面面&#xff0c;然而在昏暗场景、低照度或弱光条件下&#xff0c;图像的质量往往受到严重影响&#xff0c;表现为亮度不足、对比度低下、色彩失真以及细节丢失等问题。这类图像对于人眼识别和计算机视觉…

基于机器学习的网络入侵检测与特征选择及随机森林分类器性能评估(NSL-KDD数据集)

简介 本文将详细介绍如何利用Python和相关机器学习库对NSL-KDD数据集进行预处理&#xff0c;特征选择&#xff0c;并通过随机森林算法构建网络入侵检测模型。同时&#xff0c;还将展示如何计算并可视化模型的ROC曲线以评估其性能。 首先&#xff0c;我们导入了必要的库&#…

Java服务器-Disruptor使用注意

最近看了一下部署后台的服务器状况&#xff0c;发现我的一个Java程序其占用的CPU时长超过100%&#xff0c;排查后发现竟是Disruptor引起的&#xff0c;让我们来看看究竟为什么Disruptor会有这样的表现。 发现占用CPU时间超过100%的进程 首先是在服务器上用top命令查看服务器状…

300分钟吃透分布式缓存-23讲:Redis是如何淘汰key的?

淘汰原理 首先我们来学习 Redis 的淘汰原理。 系统线上运行中&#xff0c;内存总是昂贵且有限的&#xff0c;在数据总量远大于 Redis 可用的内存总量时&#xff0c;为了最大限度的提升访问性能&#xff0c;Redis 中只能存放最新最热的有效数据。 当 key 过期后&#xff0c;或…

专题一 - 双指针 - leetcode 202. 快乐数 | 简单难度

leetcode 202. 快乐数 leetcode 202. 快乐数 | 简单难度1. 题目详情1. 原题链接2. 基础框架 2. 解题思路1. 题目分析2. 算法原理3. 时间复杂度 3. 代码实现4. 知识与收获 leetcode 202. 快乐数 | 简单难度 1. 题目详情 编写一个算法来判断一个数 n 是不是快乐数。 「快乐数」…

geoserver+mapbox-gl 离线部署矢量切片地图服务学习笔记

geoserver安装 geoserver的安装包可以在官网下载Download - GeoServer&#xff0c;想要选择版本点击Archived找到指定版本进行下载http://geoserver.org/download/ &#xff08;如果网络不稳定&#xff0c;也可以直接使用下面的下载地址&#xff09; geoserver-2.15.0.rar资…