双指针算法——滑动窗口

前言:

滑动窗口本质上也是利用双指针来解决特定情况下的问题。滑动窗口算法思想是通过俩个指针,定义在左边和右边,俩指针同向运动,保持着一个像“窗口”一样的双指针来不停的压缩或者扩展来移动“窗口”,从而找到特定的子数组。

滑动窗口基本做题思路:

首先我们可以利用暴力解法来看看优化,是否是利用双指针来解决?双指针是否同向移动?

如果满足,可以定义为使用滑动窗口来解决,时间复杂度可以大大优化(一般为0(N))

进窗口(right指针进入,相当于扩展窗口大小,框住子数组)

判断(判断题目给定条件是否成立,要是不符合该指定条件,则left指针选择移动)

出窗口(left指针移动,相当于缩小窗口大小,直到满足题目要求即可)

更新结果(更新题目所求,得出最优结果)

一、长度最小的子数组(. - 力扣(LeetCode))

如图所述,题目要求找出一连串的子数组,其数据之和要大于等于所给定的target,而且该子数组长度是最小的,以示例一举例:

算法思路:

我们可以利用双指针来不断寻找,定义在一个“窗口”里面来找他们元素之和是否大于等于给定条件的数,先“进窗口”,right指针不断插入,遍历一遍数组,然后再判定子数组里面的元素之和是否大于等于target,若是符合,则进入“出窗口”阶段,更新最小长度,将left、继续右移,判断是否符合;若是不符合,则继续由right扩展窗口:

代码实现:

二、最大连续1的个数 (. - 力扣(LeetCode))

如图所述,给定一个数组和目标数k,题目要求是最多翻转k个0,也就是说只要不超过k次,翻转多少都可以。

相当于就是在数组中把为0的数翻转成1,最多翻转k次,然后再择出最长的连续的1的子数组,并且返回该长度。

算法思路:

首先通过暴力解法可知道双指针是同向移动,从中得知可以使用滑动窗口算法,先定义一个左右指针,然后为了不改变原数组的任何数据,我们可以使用一个计数器来统计“窗口”里面的0个数:

代码实现:

三、将x减到0的最小操作数(. - 力扣(LeetCode))

如图所述,题目要求是每一次除去左端或右端数,除去的数相加若刚好为0,则返回除去的元素数量,若无法减为0,这返回-1。

算法思路:

对于该题,常规思路不好解,代码也会很复杂。我们可以逆向思维来解题:题目告知我们需要除去左端或右端元素来解决问题,但是我们可以从中间那一段来使用滑动窗口算法解决问题:

逆向思维想好思路后利用暴力解法,我们可以再次转化为我们熟悉的滑动窗口,只需从中间选出nums_sum - (a+b) 的元素之和即可,利用“窗口”来框住子数组,剩下的就是题目所需要的最终答案了,找出目标数可以参考第一题,基本逻辑是一样的:

代码实现:

四、找到字符串中所有字母异位词(. - 力扣(LeetCode))

如图所述,题目要求是返回p中的所有异位词,异位词的定义为:

这些都称为异位词,只需要将字符串中连续的子串异位词的开始索引记录返回即可

算法思路:

依旧是使用滑动窗口来解决,但是该题毕竟不是数组,也不能使用相加的方式来做。我们可以使用哈希表来完成该题。将p中的各个字符放入哈希表中记录下来,再定义一个用于统计s中符合的子串的哈希表,最终判断俩哈希表是否相同来返回起始索引:

代码实现:

注:由于题目要求只有26个字母,所以我们可以使用数组来代替哈希表,节省时间空间复杂度

public List<Integer> findAnagrams(String s, String p) {

        List<Integer> list = new ArrayList<>();

        int[] hash1 = new int[26];//让普通数组代替hash表,26个字母

        int[] hash2 = new int[26];

        int p_len = p.length();

        int len = 0;

        for (int i = 0; i < p_len; i++) hash1[p.charAt(i) - 97]++;//创建hash1表

        for (int left=0,right=0; right < s.length();right++){

            hash2[s.charAt(right) - 97] += 1;

            len++;//进窗口

            if(len > p_len){//判断

               hash2[s.charAt(left)-97]--;

               left++;//出窗口

               len--;

            }

            if(this.My_equals(hash1,hash2)) list.add(left);//更新结果

        }

        return list;

    }

    public  boolean My_equals(int[] hash1,int[] hash2) {

        for (int i = 0; i < hash2.length; i++)

            if (hash1[i] != hash2[i])

                return false;

        return true;

    }

五、最小覆盖子串(. - 力扣(LeetCode))

如图所述,只需要包含t中所有的字符内容都可以,不论包含几个,只需要大于等于t中的字符即可,最终返回最小的子串。

算法思路:

该题与上题类似,但也有不同。区别是上题只需要统计个数即可,但此题是只论种类,不论数目.

我们依旧是用到哈希表这个容器来帮助我们完成该题。为了防止像上题一样循环26次才得出俩哈希表是否相同,我们可以使用一个小小的优化来进行改造:

然后就可以使用我们的滑动窗口来解决该问题了:

代码实现:

public String minWindow(String s, String t) {

        int[] hash1 = new int[128];//hash来代替哈希表(26+26个字母)

        int[] hash2 = new int[128];

        char in = 0, out = 0;//用来统计出入窗口时的字符

        int hash1_size = 0, len = Integer.MAX_VALUE, flg = 0;//用于统计t的种类,和最短长度,起始位置

        for (char x : t.toCharArray()) {

            hash1[x] += 1;

            if (hash1[x] == 1) hash1_size++;//只有新种类进来时才加1

        }

        for (int left = 0, right = 0, count = 0/*维护hash2的种类*/; right < s.length(); right++) {

            in = s.charAt(right);

            hash2[in]++;//进窗口

            if (hash1[in] == hash2[in]) count++;//只有相等时才加1,防止重复种类的相加

            while (count == hash1_size) {//判断

                out = s.charAt(left);

                if (len > right - left + 1) {

                    len = right - left + 1;//更新长度

                    flg = left;//更新起始位置

                }

                if (hash1[out] == hash2[out]) count--;

                hash2[out]--;

                left++;//出窗口

            }

        }

        if(len == Integer.MAX_VALUE) return "";

        else return s.substring(flg,flg + len);

    }

以上就是关于滑动窗口的经典例题,希望对大家有所帮助,谢谢各位观看!

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

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

相关文章

DigitalOcean Droplet 云主机新增内置第五代 Xeon CPU 机型

DigitalOcean 近期宣布&#xff0c;在其高级 CPU 服务器&#xff08;Premium CPU-Optimized Droplet&#xff09;队列中引入英特尔第五代Xeon可扩展处理器&#xff08;代号为 Emerald Rapids&#xff09;。作为英特尔产品线中的最新一代用于数据中心工作负载的处理器&#xff0…

深入探索PostgreSQL的复制功能:原理、配置与实践

PostgreSQL是一个高度可扩展的开源对象关系数据库系统&#xff0c;它提供了强大的数据复制功能&#xff0c;以确保数据的高可用性和容错性。PostgreSQL的复制功能允许多个数据库服务器之间同步数据&#xff0c;从而支持读写分离、负载均衡和备份等多种应用场景。本文将详细介绍…

Java接口、抽象类、继承运用练习

按以下要求编写一个完整的 Java 程序&#xff1a; &#xff08;1&#xff09;编写抽象类 GeometricObject 类&#xff1b; • 该类包括 String型的变量 color 和 boolean 型的 filled 变量&#xff1b; •该类有一个默认的构造方法&#xff1b; •两个抽象方法&#xff0c;计算…

【C/C++】静态函数调用类中成员函数方法 -- 最快捷之一

背景 注册回调函数中&#xff0c;回调函数是一个静态函数。需要调用类对象中的一个成员函数进行后续通知逻辑。 方案 定义全局指针&#xff0c;用于指向类对象this指针 static void *s_this_obj;类构造函数中&#xff0c;将全局指针指向所需类的this指针 s_this_obj this…

golang string、byte[]以及rune的基本概念,用法以及区别

在 Go 语言中&#xff0c;string、byte[] 和 rune 是处理文本和字符的三种不同数据类型。它们有各自的用途和特点&#xff0c;下面将详细介绍它们的基本概念、用法以及区别。 1. string 基本概念 字符串类型&#xff1a;string 是 Go 语言中的一种基本类型&#xff0c;用于表…

LangCell:用于细胞注释的语言-细胞预训练模型

细胞身份包括细胞的各种语义&#xff0c;包括细胞类型、pathway信息、疾病信息等。从转录组数据中了解细胞身份&#xff0c;例如注释细胞类型&#xff0c;是一项基础任务。由于语义是由人类赋予的&#xff0c;如果没有cell-label pair提供监督信号&#xff0c;AI模型很难有效地…

小项目——MySQL集训(学生成绩录入)

ddl语句 -- 创建学生信息表 CREATE TABLE students (student_id INT AUTO_INCREMENT PRIMARY KEY COMMENT 学生ID,name VARCHAR(50) NOT NULL COMMENT 学生姓名,gender ENUM(男, 女) NOT NULL COMMENT 性别,class VARCHAR(50) NOT NULL COMMENT 班级,registration_date DATE CO…

【杂记-浅谈EBGP外部边界网关协议、IBGP内部边界网关协议】

一、EBGP概述 EBGP&#xff0c;External Border Gateway Protocol&#xff0c;即外部边界网关协议&#xff0c;EBGP主要用于在不同自治系统&#xff08;AS&#xff09;之间交换路由信息&#xff0c;每个AS都有一个独特的AS号码&#xff0c;用于区分不同的自治系统。EBGP通过AS…

Redis实战—基于setnx的分布式锁与Redisson

本博客为个人学习笔记&#xff0c;学习网站与详细见&#xff1a;黑马程序员Redis入门到实战 P56 - P63 目录 分布式锁介绍 基于SETNX的分布式锁 SETNX锁代码实现 修改业务代码 SETNX锁误删问题 SETNX锁原子性问题 Lua脚本 编写脚本 代码优化 总结 Redisson 前言…

C++泛型编程之模板的使用

文章目录 1.模板的概念1.函数模板2.类模板3.模板特化4.模板元编程 2.模板的使用1.函数模板的使用2.类模板的使用3.模板特化4.模板参数推导5.普通函数和函数模板的调用规则1. 非模板函数优先2. 最佳匹配原则3. 显式模板参数指定4. 函数模板特化5. 重载决议示例 7.注意事项8.类模…

v-for中为什么要使用key

在Vue中&#xff0c;v-for指令用于循环遍历数组或对象&#xff0c;并为每个元素或属性生成相应的DOM元素或组件实例。当使用v-for循环渲染时&#xff0c;Vue会尽量复用已有的元素&#xff0c;而不是重新创建。为了实现这个复用机制&#xff0c;Vue会根据每个元素的key来跟踪它们…

基于STM32的温湿度检测TFT屏幕proteus恒温控制仿真系统

一、引言 本文介绍了一个基于STM32的恒温控制箱检测系统&#xff0c;该系统通过DHT11温湿度传感器采集环境中的温湿度数据&#xff0c;并利用TFT LCD屏幕进行实时显示。通过按键切换页面显示&#xff0c;通过按键切换实现恒温控制箱的恒温控制。为了验证系统的可靠性和稳定性&…

MongoDB的核心点是什么,选择是否使用!

MongoDB概述 定义: MongoDB是一个文档数据库&#xff0c;设计目的在于简化应用程序的开发和扩展。起源: 由DoubleClick创始人Dwight Merriman和Kevin O’Connor于2007年启动&#xff0c;以应对大规模流量需求。 MongoDB发展历程 开发背景: 由于关系型数据库无法满足DoubleCl…

在 TS 中使用 Manifold 建模

一 Manifold 是什么 1.1 简介 Manifold 是一个几何处理库&#xff0c;专注于高效、可靠的布尔运算和几何操作。它主要用于3D建模和计算几何领域&#xff0c;提供了高性能的几何算法&#xff0c;适用于需要精确几何计算的应用场景。 1.2 主要特点 高效的布尔运算&#xff1a…

Lombok的hashCode方法

Lombok对于重写hashCode的算法真的是很经典&#xff0c;但是目前而言有一个令人难以注意到的细节。在继承关系中&#xff0c;父类的hashCode针对父类的所有属性进行运算&#xff0c;而子类的hashCode却只是针对子类才有的属性进行运算&#xff0c;立此贴提醒自己。 目前重写ha…

png格式快速压缩该怎么做?在电脑压缩png图片的方法

png格式的图片如何快速压缩变小呢&#xff1f;现在网络的不断发展&#xff0c;图片是日常用来分享展示内容的一种常用手段&#xff0c;其中使用最多的一种图片格式就是png&#xff0c;png格式具有无损压缩支持透明底的特性&#xff0c;在很多的场景下都会使用。 现在图片的清晰…

本周AI动态:生成型AI的命运掌握在法院手中

本周AI领域发生了音乐公司指控两家开发AI歌曲生成器的初创公司Udio和Suno侵犯版权的事件。 美国音乐唱片行业协会&#xff08;RIAA&#xff09;周一宣布&#xff0c;由索尼音乐娱乐公司、环球音乐集团、华纳唱片公司等发起的诉讼已经提起。诉讼声称&#xff0c;Udio和Suno在未…

乒乓征途:开球网 跨越积分鸿沟的热爱与挑战

乒乓征途&#xff1a;跨越积分鸿沟的热爱与挑战 在乒乓球这项集速度、技巧与策略于一体的运动中&#xff0c;我以一名业余爱好者的身份&#xff0c;勇敢地踏上了开球网这一竞技的广阔舞台。这里&#xff0c;积分不仅是衡量实力的标尺&#xff0c;更是通往更高层次比赛的通行证…

贷款承诺状态映射参数表,用于加工的提示信息

在设计贷款承诺状态映射参数表时&#xff0c;目的是将贷款的不同状态映射为相应的提示信息&#xff0c;以便于系统能够自动生成和发送通知给相关的借款人或银行员工。以下是一个简化的参数表示例&#xff0c;用于指导贷款状态的加工和提示信息生成&#xff1a; | 状态代码 |…

Arduino - 线性执行器(支撑或滑杆)

Arduino - 线性执行器(支撑或滑杆) Arduino - 执行器 In this tutorial, we are going to learn: 在本教程中&#xff0c;我们将学习&#xff1a; How linear actuator works 线性执行器的工作原理How to make linear actuator extend or retract. 如何使线性执行器伸展或缩…