当前位置: 首页 > news >正文

「算法」滑动窗口

前言

算法需要多刷题积累经验,所以我行文重心在于分析解题思路,理论知识部分会相对简略一些

正文

滑动窗口属于双指针,这两个指针是同向前行,它们所夹的区间就称为“窗口”

啥时候用滑动窗口?

  • 题目涉及到“子序列”、“子数组”、“子串”等概念,要你求和它们相关的量,比如求满足条件的子数组的最大长度
  • 在暴力枚举的时候,如果发现两个“指针”都是朝同一个方向走的,就可以考虑滑动窗口
    注:滑动窗口可以看作是暴力枚举优化后的结果

如何使用?(步骤)

  1. 定义两个“指针”left、right(此“指针”不是真正的指针)
  2. 通过移动 right 让元素进窗口
  3. 进窗口之后进行判断,并根据这个判断决定什么时候需要出窗口

2和3需要放在一个循环里面,这样才可以让窗口不断往前走

  • 在移动“窗口”的过程我们需要不断更新结果,比如求子数组最大长度maxLen,那么每次求出一个子数组长度之后,我们就要判断是否需要更新maxLen
  • 至于是在进窗口后更新结果,还是在出窗口后更新,这就需要结合具体题目分析

例题

长度最小的子数组

思路

  1. 先用暴力枚举看看怎么做,我们先固定一个端点left,然后让right一直往右走,直到[left,right]区间中的元素和大于target,此时得到区间长度为right-left+1
  2. 既然已经比target大,那接下来就别让right往右走了,先让它停下来,因为数组中全是正数,此时right继续走的话,区间中元素总和肯定还是比target大,但是此时区间的长度肯定不是最小长度,这样就做了无用功
  3. 所以我们应该让left向前走,缩小区间,将新区间的元素和与 target 比较,若大于 target,则记录此时区间长度,并让 left 继续向前走;反之则让 right 往前走

在这里插入图片描述

上面的思路其实就是从暴力枚举逐步过渡到滑动窗口,整个过程下来,我们不难发现:砍掉暴力枚举中那些无用功,也就得到滑动窗口的思路了

代码如下:

class Solution {public int minSubArrayLen(int target, int[] nums) {int left = 0,right = 0;  //区间为左闭右闭int sum = 0;  //区间中元素总和int minLen = nums.length+1;  //最小窗口的长度,一开始初始化为一个比较大的数,不然下面使用取小函数可能没法正确更新minLenfor(;right < nums.length;right++) {sum += nums[right];while(sum >= target) {minLen = Math.min(minLen,right-left+1);  //更新 minLensum -= nums[left];left++;}}return minLen == nums.length+1 ? 0 : minLen;}
}

注意:用取小函数来更新 minLen,可以简化代码(相比于使用条件语句判断大小),同理,可以用取大函数来更新最大值

复杂度分析

时间复杂度:O(N),最坏情况下左右指针都要走完整个数组
比如数组大小为5,前四个元素都为1,最后一个为10000,target为10
空间复杂度:O(1)

相关文章:

  • 入门者拿捏 Java 的必备小秘诀
  • Linux:docker在线仓库(docker hub 阿里云)基础操作
  • VMwareWorkstation17.0虚拟机安装搭建Windows 11虚拟机(完整图文详细步骤教程)
  • Python学习路线图
  • ⭐北邮复试刷题103. 二叉树的锯齿形层序遍历 (力扣每日一题)
  • 模拟发送 Ctrl+Alt+Del 快捷键
  • 【简洁的代码永远不会掩盖设计者的意图】如何写出规范整洁的代码
  • 什么是tomcat?tomcat是干什么用的?
  • [SWPUCTF 2021 新生赛]crypto8
  • MySQL性能调优篇(3)-缓存的优化与清理
  • WordPress作者页面链接的用户名自动变成16位字符串串插件Smart User Slug Hider
  • 《小强升职记:时间管理故事书》阅读笔记
  • 【生产实测有效】Linux磁盘清理常用命令
  • 矩阵对角线元素的和
  • MySQL数据库基础(五):SQL语言讲解
  • [译]前端离线指南(上)
  • 【跃迁之路】【699天】程序员高效学习方法论探索系列(实验阶段456-2019.1.19)...
  • CEF与代理
  • HTTP那些事
  • Java知识点总结(JavaIO-打印流)
  • springboot_database项目介绍
  • Spring思维导图,让Spring不再难懂(mvc篇)
  • Vim 折腾记
  • vue-router的history模式发布配置
  • yii2中session跨域名的问题
  • Zepto.js源码学习之二
  • 免费小说阅读小程序
  • 如何合理的规划jvm性能调优
  • 如何实现 font-size 的响应式
  • 深度学习之轻量级神经网络在TWS蓝牙音频处理器上的部署
  • AI又要和人类“对打”,Deepmind宣布《星战Ⅱ》即将开始 ...
  • Redis4.x新特性 -- 萌萌的MEMORY DOCTOR
  • 关于Android全面屏虚拟导航栏的适配总结
  • 你学不懂C语言,是因为不懂编写C程序的7个步骤 ...
  • #LLM入门|Prompt#1.8_聊天机器人_Chatbot
  • (js)循环条件满足时终止循环
  • (JSP)EL——优化登录界面,获取对象,获取数据
  • (webRTC、RecordRTC):navigator.mediaDevices undefined
  • (八)五种元启发算法(DBO、LO、SWO、COA、LSO、KOA、GRO)求解无人机路径规划MATLAB
  • (超简单)使用vuepress搭建自己的博客并部署到github pages上
  • (附源码)spring boot车辆管理系统 毕业设计 031034
  • (每日持续更新)信息系统项目管理(第四版)(高级项目管理)考试重点整理第3章 信息系统治理(一)
  • (三) prometheus + grafana + alertmanager 配置Redis监控
  • (转)3D模板阴影原理
  • (转)chrome浏览器收藏夹(书签)的导出与导入
  • (转)Google的Objective-C编码规范
  • (轉貼) UML中文FAQ (OO) (UML)
  • ***微信公众号支付+微信H5支付+微信扫码支付+小程序支付+APP微信支付解决方案总结...
  • .NET Core 控制台程序读 appsettings.json 、注依赖、配日志、设 IOptions
  • .NET core 自定义过滤器 Filter 实现webapi RestFul 统一接口数据返回格式
  • .NET Micro Framework 4.2 beta 源码探析
  • .Net环境下的缓存技术介绍
  • .net使用excel的cells对象没有value方法——学习.net的Excel工作表问题
  • .net下的富文本编辑器FCKeditor的配置方法
  • .one4-V-XXXXXXXX勒索病毒数据怎么处理|数据解密恢复