详解顺序结构滑动窗口处理算法
🎀个人主页: https://zhangxiaoshu.blog.csdn.net
📢欢迎大家:关注🔍+点赞👍+评论📝+收藏⭐️,如有错误敬请指正!
💕未来很长,值得我们全力奔赴更美好的生活!
前言
在数据结构和算法方面的面试中,数组和字符串的相关问题往往是一个重要的考察点。面试官通常会测试面试者在处理这些基础数据结构时的熟练程度,因为这直接关系到解决实际问题的能力。在数组和字符串的考察中,双指针和滑动窗口以及排序算法、字符串的处理API成为关键技巧,本文主要对滑动窗口进行简单介绍。
文章目录
- 前言
- 1. 序
- 2. 滑动窗口原理
- 3. 应用场景
- (1)长度最小的子数组
- (2)无重复字符的最长子串
- (3)存在重复元素 II
- 总结
1. 序
双指针和滑动窗口是在处理数组和字符串问题时常用的技巧。双指针通常用于解决数组中的一些查找或判断问题,通过设置两个指针在数组上移动,实现对数组的遍历和比较。滑动窗口则常用于解决字符串中的子串或子数组问题,通过维护一个可变大小的窗口在字符串上滑动,从而实现对子串或子数组的探测。
排序算法在面试中同样是一个重要的考察点,因为它与数组相关,对数据的整理和查找提供了基础。熟练掌握常见的排序算法,如快速排序、归并排序等,有助于在解决各种问题时更高效地处理数组。此外,对于字符串的处理API也是面试中需要掌握的知识。熟悉字符串的各种操作,如查找子串、替换字符、反转字符串等,能够帮助面试者更灵活地处理字符串相关的问题。
本文主要是对滑动窗口这种常用技巧进行简要介绍,帮助读者在面对数组和字符串相关问题时能够更加从容应对。深入理解这些技巧,并在实际问题中灵活运用,将有助于提高面试者在数据结构和算法面试中的表现。
2. 滑动窗口原理
滑动窗口法是一种在处理数组或字符串的子序列(子数组或子串)问题时常用的技巧。它通过维护一个动态的窗口来解决问题,窗口的起始和结束位置会根据问题的要求进行滑动。这种方法通常用于求解最长子串、最短子数组等问题。
基本思路:
-
初始化窗口的起始位置和结束位置。通常使用两个指针,比如 start 和 end,表示窗口的左右边界。
-
通过移动窗口的结束位置,扩大窗口。根据问题的要求,可以通过增加 end 指针的位置来扩展窗口。
-
通过移动窗口的起始位置,缩小窗口。当窗口包含的元素满足某个条件时,可以通过增加 start 指针的位置来缩小窗口。
-
重复以上步骤直到满足问题的条件。 在每一步中,都可以根据问题的要求更新窗口的状态,并在遍历完整个数组或字符串后得到问题的解。
滑动窗口法的优势在于它能够在线性时间内解决很多子序列问题,而无需使用额外的空间。这使得它在处理大规模数据时表现良好。
3. 应用场景
(1)长度最小的子数组
给定一个含有 n 个正整数的数组和一个正整数 target 。找出该数组中满足其总和大于等于 target 的长度最小的 连续子数组 [numsl, numsl+1, …, numsr-1, numsr] ,并返回其长度。如果不存在符合条件的子数组,返回 0 。
输入:target = 7, nums = [2,3,1,2,4,3]
输出:2
解释:子数组 [4,3] 是该条件下的长度最小的子数组。
具体思路如下:
-
初始化变量 ans 为整数的最大值 Integer.MAX_VALUE,start 和 end 分别表示当前子数组的起始和结束位置,sum 表示当前子数组的和。
-
使用 while 循环遍历数组 nums,其中 end 指针负责扩大窗口,start 指针负责缩小窗口。
-
在循环中,先将当前元素加到 sum 中,然后检查是否满足 sum >= target 的条件。如果满足,说明当前窗口的子数组和大于等于目标值,此时进入内部的 while 循环。
-
在内部的 while 循环中,不断缩小窗口,即通过减去 nums[start] 的值来减小 sum。同时,更新 ans 为当前窗口的长度 end - start + 1 和 ans 之前值的较小值。这样,通过不断缩小窗口,可以找到满足条件的最小子数组。
-
重复上述过程,直到 end 指针遍历完整个数组。
-
返回最终结果 ans,如果 ans 仍然为 Integer.MAX_VALUE,说明没有找到满足条件的子数组,返回 0;否则,返回找到的最小子数组的长度。
class Solution {public int minSubArrayLen(int target, int[] nums) {int ans = Integer.MAX_VALUE;int start = 0, end = 0;int sum = 0;while(end < nums.length){sum += nums[end];while(sum >= target){sum-=nums[start];ans = Math.min(ans,end-start+1);start++;}end++;}return ans== Integer.MAX_VALUE ? 0 : ans;}
}
(2)无重复字符的最长子串
给定一个字符串 s ,请你找出其中不含有重复字符的 最长子串 的长度。
输入: s = “abcabcbb”
输出: 3
解释: 因为无重复字符的最长子串是 “abc”,所以其长度为 3。
具体思路如下:
-
初始化指针和数据结构: 使用 start 和 end 两个指针来表示当前子串的起始和结束位置,使用 HashSet 来存储当前子串中出现的字符。lenMax 用于记录最长不含重复字符的子串长度。
-
滑动窗口: 通过不断移动 end 指针,扩大窗口。当遇到重复字符时,开始缩小窗口。
-
处理重复字符:
如果当前字符是新字符
,将其添加到 set 中,然后更新 lenMax 为当前子串的长度(end - start + 1)的最大值。如果当前字符已经在 set 中
,表示有重复字符,需要将 start 指针右移,并将对应的字符从 set 中移除,直到子串中不再包含重复字符。 -
遍历完整个字符串: 通过不断移动 end 指针和更新 lenMax,直到 end 指针遍历完整个字符串。
-
返回结果: 返回最终的 lenMax,即最长不含重复字符的子串的长度。
class Solution {public int lengthOfLongestSubstring(String s) {int start=0;int end=0;HashSet<Character> set = new HashSet<Character>();int lenMax=0;while(end < s.length()){ if(set.add(s.charAt(end))){lenMax=Math.max(lenMax,end-start+1);end++; }else{set.remove(s.charAt(start));start++;}}return lenMax;}
}
(3)存在重复元素 II
给你一个整数数组 nums 和一个整数 k ,判断数组中是否存在两个 不同的索引 i 和 j ,满足 nums[i] == nums[j] 且 abs(i - j) <= k 。如果存在,返回 true ;否则,返回 false 。
输入:nums = [1,2,3,1], k = 3
输出:true
-
初始化数据结构: 使用一个 LinkedHashSet 来存储当前窗口中的元素,保持插入顺序。
-
遍历数组: 使用两个指针 start 和 end 遍历数组,其中 end 指针负责扩大窗口,start 指针负责缩小窗口。
-
判断重复元素: 在每一步中,首先检查当前窗口中是否包含数组中的元素 nums[end]。如果存在,表示存在重复元素,直接返回 true。
-
保持窗口大小: 在窗口大小达到 k 之后,通过缩小窗口,即移除 set 中的元素 nums[start],并将 start 指针右移。
-
遍历完整个数组: 重复上述步骤,直到 end 指针遍历完整个数组。
-
返回结果: 如果在遍历过程中未找到重复元素,返回 false。
class Solution {public boolean containsNearbyDuplicate(int[] nums, int k) {int numLength = nums.length;Set<Integer> set = new LinkedHashSet<>();for(int start = 0, end = 0; end < numLength; end++){if(set.contains(nums[end])){return true;}set.add(nums[end]);while (end - start >= k){set.remove(nums[start]);start++;}}return false;}
}
总结
滑动窗口算法是一种用于解决数组或字符串中子序列(子数组或子串)问题的有效技巧。它通过维护一个动态的窗口,不断调整窗口的起始和结束位置,以满足问题的条件。以下是滑动窗口算法的关键特点、优势和应用场景:
- 关键特点:
- 使用两个指针(通常是起始指针和结束指针)表示窗口的边界。
- 通过不断移动窗口的边界,动态调整窗口的大小。
- 用于解决需要求解子序列最优解的问题。
- 优势:
- 高效: 滑动窗口算法通常具有线性时间复杂度,因为每个元素或字符只需被访问一次。
- 空间效率: 使用常数级的额外空间,不需要存储整个子序列的信息,而是通过维护窗口的边界来解决问题。
- 简洁: 算法思路相对简单,易于理解和实现。
- 应用场景:
- 最长子串或子数组: 用于求解最长不含重复字符的子串、最短子数组等问题。
- 满足条件的子串: 用于找到满足特定条件的子串,如包含指定字符、和大于等于某个值的子数组等。
- 窗口内的统计信息: 用于在移动窗口的过程中动态计算窗口内元素的统计信息,如和、平均值等。
- 固定长度的窗口: 用于处理固定长度的窗口,例如计算滑动窗口的平均值。
- 注意事项:
- 确保窗口的起始和结束位置的移动是合理的,避免重复计算和漏算。
- 处理窗口的边界条件,确保不越界。
总体来说,滑动窗口算法是一种高效、简洁的解决子序列问题的方法,在处理字符串和数组等数据结构时广泛应用。
文中有不对的地方欢迎指正、补充。
相关文章:
详解顺序结构滑动窗口处理算法
🎀个人主页: https://zhangxiaoshu.blog.csdn.net 📢欢迎大家:关注🔍点赞👍评论📝收藏⭐️,如有错误敬请指正! 💕未来很长,值得我们全力奔赴更美好的生活&…...
Java 8中使用Stream来操作集合
Java 8中使用Stream来操作集合 在Java 8中,你可以使用Stream API来操作集合,这使得集合的处理变得更加简洁和函数式。Stream API提供了一系列的中间操作(intermediate operations)和终端操作(terminal operations&…...
MATLAB环境下一种改进的瞬时频率(IF)估计方法
相对于频率成分单一、周期性强的平稳信号来说,具有非平稳、非周期、非可积特性的非平稳信号更普遍地存在于自然界中。调频信号作为非平稳信号的一种,由于其频率时变、距离分辨率高、截获率低等特性,被广泛应用于雷达、地震勘测等领域。调频信…...
解决:selenium web browser 的版本适配问题
文章目录 解决方案:使用 webdriver manager 自动适配驱动 使用 selenium 操控浏览器的时候报错: The chromedriver version (114.0.5735.90) detected in PATH at /opt/homebrew/bin/chromedriver might not be compatible with the detected chrome ve…...
pytest.param作为pytest.mark.parametrize的参数进行调用
pytest.param:在 pytest.mark.parametrize 中可以作为一个指定的参数进行调用 获取数据库(网页端)数据,通过pytest.param包装成数据包用于pytest.mark.parametrize 中实现数据驱动调用。 import os import pytest import json fr…...
如何判断一个元素是否在可视区域中?
文章目录 一、用途二、实现方式offsetTop、scrollTopgetBoundingClientRectIntersection Observer创建观察者传入被观察者 三、案例分析参考文献 一、用途 可视区域即我们浏览网页的设备肉眼可见的区域,如下图 在日常开发中,我们经常需要判断目标元素是…...
Go Run - Go 语言中的简洁指令
原文:breadchris - 2024.02.21 也许听起来有些傻,但go run是我最喜欢的 Go 语言特性。想要运行你的代码?只需go run main.go。它是如此简单,我可以告诉母亲这个命令,她会立即理解。就像 Go 语言的大部分功能一样&…...
Spring全面精简总结
Spring两大核心功能:IOC控制反转、AOP面向切面的编程 控制反转(loC,Inversion of Control),是一个概念,是一种思想。指将传统上由程序代码直接操控的对象调用权交给容器,通过容器来实现对象的装配和管理。控制反转就是…...
低代码开发如何助力数字化企业管理系统平台构建
随着数字化时代的到来,企业对于管理系统的需求日益增长。高效的管理系统可以提高企业的运作效率,降低成本,提升竞争力。然而,传统的开发方式在应对日益复杂的管理系统需求时,显得力不从心。低代码开发作为一种新兴的开…...
ElasticSearch之零碎知识点
写在前面 本文记录es的零碎知识点,包括但不限于概念,集群方式,等。 1:词项查询 VS 全文查询 词项查询:查询的内容不做分词处理,输入的什么查询什么。 全文查询:查询的内容会做分词处理&…...
【春运抢票攻略浅析】
参考 最全12306放票规则,抢票策略,候补作用2023年12306抢票攻略(纯技巧) 研究放票规则,候补的时候车次进行一下挑选,能够买长乘短的尽量买长,不要候补一些区间票吧,这是一开始放票…...
【Java EE初阶二十五】简单的表白墙(一)
1. 前端部分 1.1 前端代码 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta http-equiv"X-UA-Compatible" content"IEedge"><meta name"viewport" content"wid…...
人工智能的新浪潮:探索OpenAI的Sora视频模型及其对未来创作的影响
OpenAI的最新AI视频模型Sora,自发布以来,已成为科技界的热点。Sora的核心能力在于将文本描述转化为高清视频片段,标志着在视频生成领域的一次重大突破。Sora的特点包括使用深度理解语言的能力来准确解释提示,以及生成表达丰富情感…...
【c语言】字符函数和字符串函数(上)
前言 在编程的过程中,我们经常要处理字符和字符串,为了⽅便操作字符和字符串,C语⾔标准库中提供了⼀系列库函数~ 欢迎关注个人主页:逸狼 创造不易,可以点点赞吗~ 如有错误,欢迎指出~ 目录 前言 1. 字符分…...
React18源码: schedule任务调度messageChannel
React调度原理(scheduler) 在React运行时中,调度中心(位于scheduler包)是整个React运行时的中枢(其实是心脏),所以理解了scheduler调度,就基本掌握了React的核心React两大循环:从宏…...
Jmeter 学习目录
Jmeter 所有内容均以学习为主输出内容,按照最小单位和基础进行输出。 如果有看不懂,或者有不明确的内容,欢迎大家留言说明。 Jmeter系列(1)Mac Jmeter下载安装启动 Jmeter系列(2)Jmeter 目录介…...
计算机网络 数据链路层课后题
1.以太网帧有哪些不同的封装格式?他们有何区别和应用场景? 以太网II封装(Ethernet II):以太网II封装是最常用的以太网封装格式,也被称为DIX封装。它在数据链路层首部使用6个字节的目的MAC地址和6个字节的源…...
实现验证码功能
Kaptcha 文章目录 Kaptcha介绍插件使用介绍原理引入依赖生成验证码 验证码小项目初始化前端代码约定前后端交互接口接口定义 介绍 Kaptcha 是Google的⼀个⾼度可配置的实⽤验证码⽣成⼯具 https://code.google.com/archive/p/kaptcha ⽹上有很多⼈甚⾄公司基于Google的kaptc…...
PyQt6的开发流程(密码生成小程序为例)
PyQt6的开发流程(密码生成小程序为例) 文章目录 PyQt6的开发流程(密码生成小程序为例)一、流程介绍与概览1. 界面与逻辑分离的开发流程2. PyQt6的开发流程 二、打开 designer.exe 创建文件三、用QT设计师绘制界面保存成ui1. QT常用…...
思腾云计算中心 | 5千平米超大空间,基础设施完善,提供裸金属GPU算力租赁业务
2021年,思腾合力全资收购包头市易慧信息科技有限公司,正式开启云计算业务。思腾云计算中心占地2400平米,位于包头市稀土高新区,毗邻多家知名企业,地理位置优越,交通便利,是区内重要的信息化产业…...
【Leetcode每日一题】二分查找 - 在排序数组中查找元素的第一个和最后一个位置(难度⭐⭐)(18)
1. 题目解析 Leetcode链接:34. 在排序数组中查找元素的第一个和最后一个位置 这个问题的理解其实相当简单,只需看一下示例,基本就能明白其含义了。 核心在于找到给定目标值所在的数组下标区间,设计一个O(logn)的算法。 2. 算法原…...
远程连接 vscode 出错 “远程主机可能不符合 glibc 和 libstdc++ VS Code 服务器的先决条件”
原因: vscode 版本是 1.86,服务器上的 glibc 和 libstdc 版本不满足 要求(2.28 和 3.4.25)。 解决: 1、下载 1.85.2,解压直接运行 Code.exe。 2、回退 Remote-ssh 到 0.107.1。 参考: vscode 1.86版本远程ssh不兼容旧…...
Maven入门:Java项目构建和管理的利器
Maven入门:Java项目构建和管理的利器 Maven 是一个项目管理和综合工具,它基于项目对象模型(POM)概念。Maven 可以管理项目的构建、报告和文档。以下是一篇介绍 Maven 配置和应用的教程文章。 Maven简介 Maven 使用其核心概念 POM…...
《游戏引擎架构》 -- 学习4
资源及文件系统 文件系统 游戏引擎的文件系统API通常提供以下功能: 搜需路径:是含一串路径的字符串,各路径之间以特殊字符(如冒号或分号)分隔,找文件时就会从这些路径进行搜寻。例如在命令行下执行程序&a…...
Wagtail安装运行并结合内网穿透实现公网访问本地网站界面
文章目录 前言1. 安装并运行Wagtail1.1 创建并激活虚拟环境 2. 安装cpolar内网穿透工具3. 实现Wagtail公网访问4. 固定的Wagtail公网地址 正文开始前给大家推荐个网站,前些天发现了一个巨牛的 人工智能学习网站, 通俗易懂,风趣幽默…...
10分钟快速开始SkyWalking结合Springboot项目
10分钟快速开始SkyWalking结合Springboot项目 实习期间,公司让我去学习一下链路追踪如何集成到Springboot项目中。 为此有两个方案: 1.opentelementryjaegerprometheus opentelementry 收集器收集线上的metrics和traces,然后发送给jaeger和p…...
STM32—触摸键
目录 1 、 电路构成及原理图 2 、编写实现代码 3、代码讲解 4、烧录到开发板调试、验证代码 5、检验效果 此笔记基于朗峰 STM32F103 系列全集成开发板的记录。 1 、 电路构成及原理图 触摸键简单的了解就是一次电容的充放电过程。从原理图可以看出,触摸键 …...
python中字典(dict)原理及其操作
原理 Python中的字典(Dictionary)是一种基于哈希表(Hash Table)的实现,提供了高效的键值对(Key-Value Pair)存储和访问机制。了解字典的工作原理有助于更好地理解其性能特性以及为什么在某些情…...
.NET Core Web API实现微服务集群部署
.NET Core Web API实现微服务集群部署 在.NET Core Web API中实现微服务集群部署通常涉及多个步骤,包括服务拆分、容器化、服务注册与发现、负载均衡等。以下是一个简化的步骤指南,用于在.NET Core中构建和部署微服务集群: 服…...
网络安全与信创产业发展:构建数字时代的护城河
✨✨ 欢迎大家来访Srlua的博文(づ ̄3 ̄)づ╭❤~✨✨ 🌟🌟 欢迎各位亲爱的读者,感谢你们抽出宝贵的时间来阅读我的文章。 我是Srlua,在这里我会分享我的知识和经验。&#x…...
临沂教育平台网站建设/网页seo搜索引擎优化
文章目录前言一、二叉树的遍历1.1 创建二叉树1.2 二叉树的遍历方式1.3 二叉树遍历的实现二、二叉树的其他操作前言 如果你还不知道树及二叉树的概念,请先看这篇文章树和二叉树的介绍 对于二叉树,我们学习的重点是二叉树的结构,而想要学好二…...
网站头部图片如何做/电子商务网站建设案例
众所周知,类的对象会随着程序的终止而被垃圾收集器销毁。如果要在不重新创建对象的情况下调用该类,该怎么做?这就可以通过序列化将数据转换为字节流。 对象序列化是一个用于将对象状态转换为字节流的过程,可以将其保存到磁盘文件中…...
做防护用品的网站/促销活动推广语言
详细参考:http://blog.csdn.net/taiyang1987912/article/details/41869181转载于:https://blog.51cto.com/poseidon2011/1913219...
学校网站建设方案论文/网站权重是什么意思
我们不常用但是比较实用的SHELL脚本:1、apt-get install sl 这时候忘记了sudo ,那么只要执行sudo !!,!!表示上一条命令2、vi hello.c 退出后,如果想打开刚才的hello.c文件,只需要!vi3、man ascii 来查看ascii码表4、ec…...
网站建设的7种流程图/semi final
转载于:https://www.cnblogs.com/zhongshujunqia/p/4660853.html...
有什么做兼职的医疗网站/专门用来查找网址的网站
这与其他一些问题有关,如:this,还有一些其他问题。在this question和其他人中,我们看到我们可以在一个很好的步骤中声明和初始化字符串数组,例如:const char* const list[] {"zip", "zam&q…...