算法刷题笔记 滑动窗口(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;
}

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

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

相关文章

用HttpURLConnection复现http响应码405

目录 使用GET方法&#xff0c;访问GET接口&#xff0c;服务端返回405使用GET方法&#xff0c;访问POST接口&#xff0c;服务端返回405使用POST方法&#xff0c;访问GET接口&#xff0c;服务端返回405 使用GET方法&#xff0c;访问GET接口&#xff0c;服务端返回405 发生场景&a…

Linux shell编程学习笔记63:free命令 获取内存使用信息

0 前言 在系统安全检查中&#xff0c;内存使用情况也是一块可以关注的内容。Linux提供了多个获取内存信息的命令很多。今天我们先研究free命令。 1 free命令的功能、用法和选项说明 1.1 free命令的功能 free 命令可以显示系统内存的使用情况&#xff0c;包括物理内存、交换…

Java多语言跨境电商外贸商城源码 tiktok商城系统源码 跨境电商源码

Java多语言跨境电商外贸商城源码 tiktok商城系统源码 跨境电商源码 技术栈 PC端使用&#xff1a;vueelementui 用户端使用&#xff1a;uniapp 管理端使用&#xff1a;vueelementui 后台服务使用&#xff1a;springbootmybatisplusmysql 功能描述&#xff1a; 对接PayPal…

vue3+electron项目搭建,遇到的坑

我主要是写后端,所以对前端的vue啊vue-cli只是知其然,不知其所以然 这样也导致了我在开发前端时候遇到了很多的坑 第一个坑, vue2升级vue3始终升级不成功 第二个坑, vue add electron-builder一直卡进度,进度条走完就是不出提示succes 第一个坑的解决办法: 按照网上说的升级v…

Ubuntu 20.04下多版本CUDA的安装与切换 超详细教程

目录 前言一、安装 CUDA1.找到所需版本对应命令2.下载 .run 文件3.安装 CUDA4.配置环境变量4.1 写入环境变量4.2 软连接 5.验证安装 二、安装 cudnn1.下载 cudnn2.解压文件3.替换文件4.验证安装 三、切换 CUDA 版本1.切换版本2.检查版本 前言 当我们复现代码时&#xff0c;总会…

深入分析SSL/TLS服务器的证书(C/C++代码实现)

SSL&#xff08;Secure Sockets Layer&#xff09;和TLS&#xff08;Transport Layer Security&#xff09;是网络安全领域的重要协议&#xff0c;它们在保护网络通信中发挥着至关重要的作用。这些协议通过加密和身份验证机制&#xff0c;确保数据在传输过程中的机密性和完整性…

初见:AntDB智能运维“三剑客“之ACC

前情回顾 在前两个章节中&#xff0c;我们介绍了 AntDB 智能运维"三剑客"的 ADC 和 MTK。 初见&#xff1a;AntDB智能运维"三剑客"之ADC 初见&#xff1a;AntDB智能运维"三剑客"之MTK 本文将继续介绍 AntDB 数据库智能运维平台 ACC。 AntDB 介绍…

minist数据集分类模型的训练

minist数据集训练 训练方法&#xff1a;利用pytorch来实现minist数据集的分类模型训练 训练模型如下图所示 模型代码&#xff1a; import torch from torch import nn from torch.nn import Flattenclass Net(nn.Module):def __init__(self):super().__init__()self.module …

ChatGPT对话:Scratch编程中一个单词,如balloon,每个字母行为一致,如何优化编程

【编者按】balloon 7个字母具有相同的行为&#xff0c;根据ChatGPT提供的方法&#xff0c;优化了代码&#xff0c;方便代码维护与复用。初学者可以使用7个字母精灵&#xff0c;复制代码到不同精灵&#xff0c;也能完成这个功能&#xff0c;但不是优化方法&#xff0c;也没有提高…

FairJob:促进在线广告系统公平性研究

在人工智能&#xff08;AI&#xff09;与人类动态的交汇处&#xff0c;既存在机遇也存在挑战&#xff0c;特别是在人工智能领域。尽管取得了进步&#xff0c;但根植于历史不平等中的持续偏见仍然渗透在我们的数据驱动系统中&#xff0c;这些偏见不仅延续了不公平现象&#xff0…

Centos新手问题——yum无法下载软件

起因&#xff1a;最近在学习centos7&#xff0c;在VM上成功安装后&#xff0c;用Secure进行远程登陆。然后准备下载一个C编译器&#xff0c;看网络上的教程&#xff0c;都是用yum来下载&#xff0c;于是我也输入了命令&#xff1a; yum -y install gcc* 本以为会自动下载&…

docker部署redis/mongodb/

一、redis 创建/root/redis/conf/redis.conf 全部执行命令如下 docker run -it -d --name redis -p 6379:6379 --net mynet --ip 172.18.0.9 -m 400m -v /root/redis/conf:/usr/local/etc/redis -e TXAsia/Shangehai redis redis-server /usr/local/etc/redis/redis.conf 部署…

C#——密封类详情

密封类 密封类是密封方法的扩展&#xff0c;用于确保某个类不会被继承。在C#中&#xff0c;你可以使用sealed关键字来声明一个密封类。 public sealed class SealedClass {// 类成员定义 } 如果使用密封类继承的话&#xff0c;程序会报错&#xff01;&#xff01;&#xff0…

01:spring

文章目录 一&#xff1a;常见面试题1&#xff1a;什么是Spring框架&#xff1f;1.1&#xff1a;spring官网中文1.2&#xff1a;spring官网英文 2&#xff1a;谈谈自己对于Spring IOC和AOP的理解2.1&#xff1a;IOCSpring Bean 的生命周期主要包括以下步骤&#xff1a; 2.2&…

解决微信读书和Apple Books导入epub电子书不显示图片的问题

title: 解决微信读书和Apple Books导入epub电子书不显示图片的问题 tags: 个人成长 categories:杂谈 最近找到一本很喜欢的书的电子版的epub版&#xff0c;发现无论是导入微信读书&#xff0c;还是Apple家的Books, 都无法正常显示图片。 于是我用calibre打开epub电子书&#x…

安卓虚拟位置修改

随着安卓系统的不断更新&#xff0c;确保软件和应用与最新系统版本的兼容性变得日益重要。本文档旨在指导用户如何在安卓14/15系统上使用特定的功能。 2. 系统兼容性更新 2.1 支持安卓14/15&#xff1a;更新了对安卓14/15版本的支持&#xff0c;确保了软件的兼容性。 2.2 路…

linux中可执行文件为什么不能拷贝覆盖

对于一个普通的文件&#xff0c;假如有两个文件&#xff0c;分别是file和file1&#xff0c;我们使用 cp file1 file的方式使用file1的内容来覆盖file的内容&#xff0c;这样是可以的。 但是对于可执行文件来说&#xff0c;当这个文件在执行的时候&#xff0c;是不能通过cp的方…

将 KNX 接入 Home Assistant 之四 功能测试

一&#xff1a;测试标准的KNX网关功能 测试软件识别是否正常 可以看到再ETS6和ETS5上都能正常识别 测试数据收发 可以正常发送数据 测试配置设备参数&#xff08;下载配置&#xff09; 可以看出&#xff0c;在ETS5上是可以正常下载参数的 但是 ETS6下载是失败的&#xff…

Pandas 学习笔记(四)--CSV文件

CSV文件 CSV&#xff08;Comma-Separated Values&#xff0c;逗号分隔值&#xff0c;有时也称为字符分隔值&#xff0c;因为分隔字符也可以不是逗号&#xff09;&#xff0c;其文件以纯文本形式存储表格数据&#xff08;数字和文本&#xff09;。 读取与写入 读取csv文件 i…

可视化作品集(08):能源电力领域

能源电力领域的可视化大屏&#xff0c;有着巨大的用武之地&#xff0c;不要小看它。 监控能源生产和消耗情况&#xff1a; 通过可视化大屏&#xff0c;可以实时监控能源生产和消耗情况&#xff0c;包括发电量、能源供应情况、能源消耗情况等&#xff0c;帮助管理者及时了解能…