算法刷题笔记 滑动窗口(C++实现,非常详细)

文章目录

    • 题目描述
    • 基本思路
    • 实现代码

题目描述

  • 给定一个大小为n ≤ 10^6的数组。
  • 有一个大小为k的滑动窗口,它从数组的最左边移动到最右边。
  • 你只能在窗口中看到k个数字。
  • 每次滑动窗口向右移动一个位置。以下是一个例子:
    • 该数组为 [1 3 -1 -3 5 3 6 7],k为3
      在这里插入图片描述
  • 你的任务是确定滑动窗口位于每个位置时,窗口中的最大值和最小值。

输入格式

  • 输入包含两行。
  • 第一行包含两个整数nk,分别代表数组长度和滑动窗口的长度。
  • 第二行有n个整数,代表数组的具体数值。同行数据之间用空格隔开。

输出格式

  • 输出包含两个。
  • 第一行输出,从左至右,每个位置滑动窗口中的最小值。
  • 第二行输出,从左至右,每个位置滑动窗口中的最大值。

基本思路

这道题是单调队列系列题目的最基础的模板题,但是对于像我这样的初学者来说仍然难度较大,因此我将对该题的思路进行详细解析。

  • 算法题的核心是将一个有直观、暴力的思路的算法优化为一个逻辑上更加复杂,但是时间和空间上更占优势的算法。那么,对于这道题,我们就需要首先思考蛮力算法的求解步骤是什么。本题的蛮力算法非常直观,就是一个简单的双重遍历。例如,对于数组[1 3 -1 -3],并假设窗口大小为3,那么我们就从数组的第一个元素开始遍历,向右滑动窗口,每次遍历到的数组元素作为窗口的左端点。以这个例子为例,可以得到两个窗口[1 3 -1][3 -1 -3],再分别从这两个窗口中找出最大值和最小值即可。代码实现上,可以大致如下:
#include <cstdio>
#include <vector>
using namespace std;

// 本例子中n为4,k为3,假设原始数组为arr,本例子以最大值为例
for(int i = 0; i < n - k + 1; ++ i)
{
    // 使用一个向量表示当前窗口,并向该窗口中添加元素
    vector<int> window;
    for(int j = i; i < i + k; ++ j) window.push_back(arr[j]);
    // 查找该向量中的最大值并输出
    int max = window[0];
    for(int j = 1; j < window.size(); ++ j) if(window[j] > max) max = window[j];
    printf("%d ", max);
}
  • 但是,我们仔细考虑一下,这种方法存在明显的冗余性。两个相邻的滑动窗口之间,有且只有一个元素不相同,而窗口中的其他元素都是完全一样的,因此会经过多轮重复遍历,创建多个大部分元素都相同的向量,算法的时间复杂度为O(nk)。既然存在冗余元素,那么我们就需要从数据结构和算法的角度上考虑对算法进行优化。
  • 直观上,我们可以发现既然相邻的两个窗口只有一个元素存在区别,即相当于下一个窗口的元素是去除了上一个窗口中的首元素,并且在后面添加了一个新元素,这就很类似于数据结构中常用的队列数据结构。因此,如果能够使用队列来代替向量,那么就可以提高算法的效率。实现代码如下:
#include <cstdio>
#include <deque>
using namespace std;

// 首先创建一个队列,并以第一个窗口中的元素进行初始化
deque<int> window;
for(int i = 0; i < k; ++ i) window.push_back(arr[i]);
// 每轮遍历队首元素出栈,并从队尾入队一个元素
for(int i = k; i < n - k + 1; ++ i)
{
    window.pop_front();
    window.push_back(arr[i]);
    // 查找当前队列中的最大值
    int max = window.front();
    for(int item : window) if(item > max) max = item;
    printf("%d ", max);
}
  • 基于队列的实现代码的确能够有更高的时间效率,但是是否可以进一步优化呢?我们发现,尽管使用队列可以更加方便地创建和维护一个窗口,而不像向量那样需要每次完全重新新建一个,但是在查找最大值时,仍然需要遍历整个队列。如果我们想要继续提高效率,就必须简化查找过程,避免耗时的循环遍历,此时就应该使用单调队列进行处理。

  • 仍然以[1 3 -1 -3]为例,当我们每一轮循环更新队列时,我们可以修改我们的更新策略。下面进行举例说明,以查找最大值为例。

    • 第一轮:队列初始为空,因此直接将第一个元素1放入队尾即可。
    • 第二轮:队列目前为[1],当前遍历到的元素为3;由于3大于当前的队尾元素1,因此如果3也放入队尾后,在查找最大值的过程中,元素1一定不会成为任何一个窗口的最大值了。这是因为当元素1和元素3在同一个窗口中时,31大,因此最大值不可能是1;当元素1和元素3不在同一个窗口中时,只有一种可能,就是当前窗口中已经不包含1了,这是因为3在原始数组中排在1的后面,只要1在窗口中,3一定在窗口中,所以这种情况下,窗口中已经不包含有元素1,所以自然不会成为最大值。因此,可以认为队列中的1为冗余元素,可以直接将其出队。只有某个元素可能成为某个窗口的最大值时,才会被放入队尾进入队列中,而所有确定下来的冗余元素都出队。所以,在第二轮迭代中首先通过上述比较过程,让队尾的1出队,此时队列为空,则直接把当前元素3放入队列中。
    • 第三轮:队列目前为[3],当前遍历到的元素是-1。由于队尾元素3-1更大,因此直接将-1入队放入队尾即可,这是因为3会在-1之前从队头离开队列,此时-1就有可能成为某个窗口的最大值元素。
    • 第四轮:和第三轮类似,队列目前是[3 -1],当前遍历到的元素是-3。由于队尾元素-1-3更大,因此直接将-3放入队尾。
  • 那么,应该如何确定何时要将队头元素出队呢?队头元素出队表示该元素已经不在当前的窗口中,最简单的处理方法就是用另一个队列记录所有队列中的元素在数组中的下标,并在每一轮的遍历过程中,通过下标判定队首元素是否在窗口中即可。下标队列和元素队列中的元素应该是一一对应的,需要同时添加和同时删除。

实现代码

#include <cstdio>
#include <deque>
using namespace std;

const int N = 1e6 + 10;
int arr[N];

int main(void)
{
    int n, k;
    scanf("%d%d", &n, &k);
    for(int i = 0; i < n; ++ i) scanf("%d", &arr[i]);
    deque<int> q;
    deque<int> index;
    // 最小值部分
    for(int i = 0; i < n; ++ i)
    {
        while(!q.empty() && q.back() >= arr[i])
        {
            q.pop_back();
            index.pop_back();
        }
        q.push_back(arr[i]);
        index.push_back(i);
        if(i >= k && i - k == index.front())
        {
            q.pop_front();
            index.pop_front();
        }
        if(i > k - 2) printf("%d ", q.front());
    }
    // 清空两个队列,对称地求解最大值
    q.clear();
    index.clear();
    printf("\n");
    for(int i = 0; i < n; ++ i)
    {
        while(!q.empty() && q.back() <= arr[i])
        {
            q.pop_back();
            index.pop_back();
        }
        q.push_back(arr[i]);
        index.push_back(i);
        if(i >= k && i - k == index.front())
        {
            q.pop_front();
            index.pop_front();
        }
        if(i > k - 2) printf("%d ", q.front());
    }
    return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

leetcode 66. 加一

leetcode 66. 加一 题解 刚开始只是以为在最后一位上加一就可以了 &#xff0c; 没想到还有进位呢&#xff0c; 比如说9的话&#xff0c; 加上1就是10&#xff0c; 返回的数组就是[1. 0],把进位的情况考虑进去就可以了。 class Solution { public:vector<int> plusOne(…

Vue3+.NET6前后端分离式管理后台实战(二十八)

1&#xff0c;Vue3.NET6前后端分离式管理后台实战(二十八)

Raw Socket(一)实现TCP三次握手

实验环境&#xff1a; Windows物理机&#xff1a;192.168.1.4 WSL Ubuntu 20.04.6 LTS&#xff1a;172.19.32.196 Windows下的一个http服务器&#xff1a;HFS&#xff0c;大概长这个样子&#xff1a; 客户端就是Ubuntu&#xff0c;服务端就是这个…

[图解]SysML和EA建模住宅安全系统-12-内部块图

1 00:00:00,580 --> 00:00:02,770 接下来我们来画流了 2 00:00:03,100 --> 00:00:05,050 首先第一个是站点状态 3 00:00:05,140 --> 00:00:08,130 从这里到这里&#xff0c;我们画一个过来 4 00:00:10,290 --> 00:00:11,890 这里流到这里 5 00:00:11,900 -->…

多粒度封锁-封锁粒度、多粒度封锁模式

一、引言 1、若采用封锁技术实现并发控制&#xff0c;事务在访问数据库对象前要在数据库对象上加锁&#xff0c;为提高事务的并发程度&#xff0c;商用DBMS会采用一种多粒度封锁方法 2、事务可访问的数据库对象可以是逻辑单元&#xff0c;包括关系、关系中的元组、关系的属性…

Python学习笔记31:进阶篇(二十)pygame的使用之图形绘制

前言 基础模块的知识通过这么长时间的学习已经有所了解&#xff0c;更加深入的话需要通过完成各种项目&#xff0c;在这个过程中逐渐学习&#xff0c;成长。 我们的下一步目标是完成python crash course中的外星人入侵项目&#xff0c;这是一个2D游戏项目。在这之前&#xff…

Debezium报错处理系列之第111篇:Can‘t compare binlog filenames with different base names

Debezium报错处理系列之第111篇:Cant compare binlog filenames with different base names 一、完整报错二、错误原因三、解决方法Debezium从入门到精通系列之:研究Debezium技术遇到的各种错误解决方法汇总: Debezium从入门到精通系列之:百篇系列文章汇总之研究Debezium技…

论文辅助笔记:ST-LLM

1 时间嵌入 2 PFA&#xff08;Partial Frozen Architecture&#xff09; 3 ST_LLM 3.1 初始化 3.2 forward

蓝桥杯开发板STM32G431RBT6高阶HAL库学习FreeRtos——FreeRTOS任务调度方式

一、任务调度方式 1.1抢占式调度&#xff08;不同优先级&#xff09; 主要是针对优先级不同的任务&#xff0c;每个任务都有一个优先级&#xff0c; 优先级高的任务可以抢占优先级低的任务。1.2时间片调度&#xff08;同优先级&#xff09; 主要针对优先级相同的任务&#x…

视频技术助力智慧城市一网统管:视频资源整合与智能化管理

随着信息技术的飞速发展&#xff0c;智慧城市已成为现代城市发展的重要方向。在智慧城市建设中&#xff0c;一网统管作为城市管理的重要策略&#xff0c;通过整合各类信息资源&#xff0c;实现资源的优化配置和问题的快速响应。其中&#xff0c;视频技术作为一网统管场景中的关…

【Linux系统】动态库和静态库 动态库加载

认识动态库静态库 我们有没有使用过库呢&#xff1f;-- 用过c、c的标准库 c的各种函数&#xff0c;c的各种STL容器&#xff0c;我们使用他们内部必须得有具体实现。 Linux: .so(动态库) .a(静态库) Windows: .dll(动态库) .lib(静态库) 库是拿来给别人使用的&#xff0c;所…

sd调试记录:

现象&#xff1a;SDIO读取TF卡&#xff0c;&#xff11;bit模式正常&#xff08;一切操作都正常&#xff09;&#xff0c;4bit模式无法读取&#xff1a; 比如在使用函数f_opendir(&DirInf, SDPath)、f_open(&SDFile, path, FA_CREATE_ALWAYS | FA_WRITE)函数时会出现错…

局部静态变量实现的单例存在多个对象

文章目录 背景测试代码运行测试尝试打开编译器优化进一步分析 背景 业务中出现日志打印失效&#xff0c;发现是因为管理日志对象的单例在运行过程中存在了多例的情况。下面通过还原业务场景来分析该问题。 测试代码 /* A.h */ #ifndef CALSS_A #define CALSS_A#include <…

如何从 Windows 11/10/8.1/8/7 恢复已删除的视频

意外删除了视频或格式化了 SD 卡/硬盘&#xff1f;没有备份已删除的视频&#xff1f;别担心&#xff0c;我们有解决方案来恢复 Windows 11、10 中已删除的视频并处理这种糟糕的情况。 但在了解如何恢复已删除的视频和视频恢复应用程序之前&#xff0c;请知道 Windows 会为您提…

IDEA与通义灵码的智能编程之旅

1 概述 本文主要介绍在IDEA中如何安装和使用通义灵码来助力软件编程,从而提高编程效率,创造更大的个人同企业价值。 2 安装通义灵码 2.1 打开IDEA插件市场 点击IDEA的设置按钮,下拉选择Plugins,如下: 2.2 搜索通义灵码 在搜索框中输入“通义灵码”,如下: 2.3 安…

【雷丰阳-谷粒商城 】【分布式高级篇-微服务架构篇】【17】认证服务01—短信/邮件/异常/MD5

持续学习&持续更新中… 守破离 【雷丰阳-谷粒商城 】【分布式高级篇-微服务架构篇】【17】认证服务01 环境搭建验证码倒计时短信服务邮件服务验证码短信形式&#xff1a;邮件形式&#xff1a; 异常机制MD5参考 环境搭建 C:\Windows\System32\drivers\etc\hosts 192.168.…

rsyslog日志转发

前言 Rsyslog可用于接受来自各种来源(本地和网络)的输入&#xff0c;转换它们&#xff0c;并将结果输出到不同&#xff08;通过模板和filter过滤&#xff09;的目的地&#xff08;目录文件中&#xff09; rsyslog是一个开源工具&#xff0c;被广泛用于Linux系统以通过TCP/UDP…

树莓派5安装冬瓜HAOS教程

原文来自瀚思彼岸和hasshome 一、安装前准备 &#xff08;1&#xff09;软件 1、树莓派烧录软件Imager 2、冬瓜HAOS镜像 &#xff08;2&#xff09;硬件 1、树莓派5 2、TF卡&#xff08;SanDisk Extreme PRO 64GB U3 A2 V30 4k&#xff09; 3、读卡器 4、键盘和鼠标 5、显…

550kg级大载重长航时无人机直升机技术详解

550kg级大载重长航时无人机直升机&#xff0c;作为一种高性能的无人机系统&#xff0c;具备了多项先进的技术特点&#xff0c;以满足高海拔、高寒等复杂环境下的应用需求。这些无人机直升机通常具备高载重、长航时、强适应性、高可靠性和良好的任务拓展性。 设备由无人直升机平…

ctfshow-web入门-文件上传(web151-web160)

目录 1、web151 2、web152 3、web153 4、web154 5、web155 6、web156 7、web157 8、web158 9、web159 10、web160 1、web151 试了下前端只能传 png 后缀的 将一句话木马改成 png 后缀&#xff0c;上传后用 burpsuite 抓包 绕过前端检测后&#xff0c;改回 php 后缀&am…