leetcode 力扣刷题 滑动窗口 部分题解(记录)

news2025/1/6 9:58:58

力扣刷题 滑动窗口相关的部分题解

  • 209. 长度最小的子数组
  • 904. 水果成篮
  • 76. 最小覆盖子串

209. 长度最小的子数组

leetcode题目链接 209.长度最小的子数组
题目内容是这样的:给定一个含有 n个正整数的数组和一个正整数 target
找出该数组中满足其和 ≥ target长度最小连续子数组 [numsl, numsl+1, …, numsr-1, numsr] ,并返回其长度。如果不存在符合条件的子数组,返回 0。
理解题意,是要找到这样的连续子数组,满足条件①子数组元素之和≥target;②目标子数组是所有满足条件①的数组中,所含元素个数最少的。 这里强调连续子数组,是因为可能存在题目要求——找到数组中的最少的几个元素,使得元素之和≥target。
寻找连续子数组,可以使用最简单的暴力求解,两遍循环找到答案,时间复杂度O(N^2)。
在暴力求解中,很多子集的寻找是没有必要的,比如下面的情况:子数组[3, 1, 2 ,4]的和已经满足≥target,题意中说明了是正整数数组,那么再往后增加元素,子数组之和会继续增大(一定满足题意),但是子数组的长度也会增大,然而我们的目标是找到满足和≥target的长度最短的子数组。因此在一个子数组之后满足≥target后,固定i,j继续增大是没有必要的。
在这里插入图片描述
这道题目可以使用滑动窗口解决,时间复杂度O(N)。思路:

  • 1、首先两个指针left和right,分别表示子数组的边界;
  • 2、固定left,right向右移动,寻找到和≥target的子数组后停止! ——此时的子数组之和是从nums[left]+……+nums[right],是从左往右累加达到了目标,没有nums[right]子数组的和是<target的,但是没有nums[left]呢?那就不一定啦~ ——因此,接下来就是在这个满足了≥target要求的可行解上,寻找最优解
  • 3、固定right,left向右移动。逐步缩小子数组的长度的同时,验证子数组和是否满足≥target。left一直向右移动,直到子数组和<target,那么此时的left向左一位就是在这个子数组中满足条件的最短的子子数组了。
  • 4、此时的left和right之间的数组不满足≥target了,并且在right之前的子数组都探索了,因此重复2-3,遍历完整个数组。
    代码如下(C++):
class Solution {
public:
    int minSubArrayLen(int target, vector<int>& nums) {
        int n = nums.size();
        int ans = INT_MAX;
        int start = 0, end = 0, sum = 0;
        while(end < n){
            sum+=nums[end];
            while(sum>=target){
            	//满足循环要求(题目要求)进入循环,判断当前子数组长度和ans哪个更短,更新
                ans = end - start + 1 < ans ? end - start + 1 : ans;
                sum -= nums[start];//移除掉start,即left元素,尝试缩短子数组的长度,寻找最优解
                start++;
            }
            end++;//end,即right要一直向右移动到知道sum>=target才会进入下面的循环
        }
        //注意最后可能没有解,所以子数组长度是0
        return  ans == INT_MAX ? 0 : ans;
    }
};

904. 水果成篮

leetcode题目链接 904. 水果成篮
在这里插入图片描述
理解题意:①只有两个篮子,那么就只能采摘两种水果;②选择一棵树开始,比如是fruits[left],那么从这颗树开始,每棵树摘一颗果子,直到遇到第三种果树停止,假设是fruits[right];③要寻找right - left + 1最大(摘得的果子数最多)的情况。
简化题意:只包含两种果树的最长连续子数组
解题思路(滑动窗口):

  • 1、用left、right表示子数组的下标,left=0,right=0;kindleft = fruits[left]表示第一种水果,kindright = fruits[right]表示第二种水果;先固定left不变,right向右移动;
  • 2、如果fruits[right]是left~right-1子数组的两种元素之一,即两种水果kindleft、kindright之一,right就继续向右移动;过程中更新ans;
  • 3.1、如果如果fruits[right]既不是kindleft也不是kindright,即出现了第三种水果,那么当前的子数组的两个元素应该更新。更新为哪两种呢?①可以肯定的是fruits[right]是一种新水果;②另外一种是fruits[right-1](可能是kindleft也可能是kindright)——因为fruits[right] ≠ kindleft && fruits[right] ≠ kindright,但是fruits[right] =kindleft || fruits[right] = kindright,可以得到fruits[right] ≠ fruits[right-1],而子数组是连续的且只能包含两种元素,因此就是fruits[right] 和 fruits[right-1]。
  • 3.2 更新子数组中包含的两种新元素后,新的子数组长度也需要更新,即更新left。left从right-1向左移动,直到找到不等于fruits[right-1]的下标(因为子数组中只能包括两种元素)。
    代码(C++):
class Solution {
public:
    int totalFruit(vector<int>& fruits) {
    	//记录子数组下标
        int left = 0, right = 0;
        //记录两种水果——即子数组中的两种元素
        int kindleft = fruits[left], kindright = fruits[right];
        //答案——水果数量
        int ans = 0;
        while(right < fruits.size()){
        	//right向右移动,如果是left~right-1子数组中的元素,并入子数组
            if(fruits[right] == kindright || fruits[right] == kindleft){
                ans = max(ans, right - left + 1 );
                right++;
            }
            else{//如果fruits[right]是第三种水果,更新kindleft、kindright
               kindright = fruits[right];
               left = right - 1;
               kindleft = fruits[left];
               //更新left,找到并入了新水果品种后的子数组的左边界
               while(left>=1 && fruits[left-1] == kindleft) left--;
               ans = max(ans, right - left + 1);
           } 
        }
        return ans;
    }
};

76. 最小覆盖子串

leetcode题目链接 76. 最小覆盖子串
题目内容如下:在这里插入图片描述
理解题意是寻找s中的最小子串,该子串能够包含t中的全部元素,根据示例可以看出,t中元素的顺序是不重要的;对于重复元素在子串中出现次数不少于在t中重复次数。
问题:在不要求字符顺序的前提下,如果确定子串substring中包含了t中全部元素?—— 对比 字符频数。即统计字符串t中字符及其频数,然后substring更新的过程中也记录子串中包含的t中元素的频数。 直到substring中包含了t中全部元素,并且对应频数≥t中对应元素的频数。 代码实现用distance表示两者差异,substring中新增一个t中的元素时,distance - 1,直到distance = 0,表示子串substring中包含了t中全部元素。
滑动窗口解题思路:

  • 1、子串的左右下标用left、right表示。先寻找到一个包含了t中全部元素的子串,固定left,right向右移动。移动过程中如果s[right]是t中元素,那么就统计其频数,如果此时该元素频数已经>t中该元素频数,那么该元素的累积对减少两者的差异没有作用了,即distance不变化;如果该元素的频数 ≤ t中该元素的频数,distance - 1;如果s[right]不是t中元素,直接right++;
  • 2、直到distance == 0时,找到了包含t中全部元素的子串,然后在子串中寻找最优解。固定right,left向右移动,left++。如果s[left]是t中元素,删除s[left]后,对应子串中关于s[left]的频数-1,如果减一后 ≥ t中该元素的频数,不会改变此时子串仍然是包含t中全部元素的事实,更新left,同时更新ans;如果如果减一后<t中该元素的频数,那么此时子串与t中元素的差异增加一个,distance++;
  • 3、重复1-2,直到遍历完s。
    代码如下(C++):
class Solution {
public:
    string minWindow(string s, string t) {
        int left = 0, right = 0;//记录子串的在S中的左右下标        
        unordered_map<int,int> T_freq, Sub_freq; //记录t中元素及其频数,记录子串中对应t中元素的频数        
        int distance = t.size(); //distance表示子串中包含的t中对应元素(及数量)与t的差异        
        int ans_left = 0, ans_right = s.size() -1; //记录最终目标子串的左右下标        
        int flage = 0;  //记录s中是否存在包括t中全部元素的子串        
        for(int i = 0; i < t.size(); i++) T_freq[t[i]]++;  //统计t中元素及其频数
        while(right < s.size()){ //遍历s 先固定left,right向右移动
        	//如果s[right]是t中的元素,将其统计在子串的频数中
            if(T_freq.count(s[right]) != 0){
                Sub_freq[s[right]]++;
                //如果还没有包含完t中对应元素,那么新增一个s[right],子串和t的差异-1
                //如果已经包含了,那么s[right]只是在频数上有累计,但是子串和t的差异不变
                if(Sub_freq[s[right]] <= T_freq[s[right]])
                    distance--;
            }
            //如果distance=0,即子串中已经包含了t中全部元素,固定right,left向右移动,缩短子串长度寻找最优解
            while(!distance){
                flage = 1;//一旦distance=0,即存在满足条件的子串,不会返回空串""
                //更新结果
                if(ans_right - ans_left > right - left) {
                    ans_right =  right;
                    ans_left = left;
                }
                //如果删除的s[left]不是t中元素,直接删除,不会影响子串中包含了t中全部元素的事实
                //如果删除的s[left]是t中元素
                if(T_freq.count(s[left]) !=0 ){
                    Sub_freq[s[left]]--;//需要更新子串中对应s[left]的频数
                    //更新频数后可能还是包含t中全部元素的,比如t中a有2个,子串中a有4个,s[left]=a,之后子串中有3个a,但是还是包含了t中全部元素
                    //如果不能全部包含了,子串和t的差异+1;
                    if(Sub_freq[s[left]]<T_freq[s[left]])
                        distance++;
                }
                left++; //删除s[left]后left右移;
            }
            right++;
        }
        if(flage) return s.substr(ans_left , ans_right - ans_left + 1);
        else return "";
    }
};

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

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

相关文章

不同路径 II——力扣63

class Solution {public:int uniquePathsWithObstacles(vector<vector<int>>& obstacleGrid) {int n=

Java中处理表格

大家好 , 我是苏麟 , 也是很久没有更新了 , 今天带来一个很好使用的一个库 easyexcel. JAVA后端开发中可能会遇到一个问题 , 就是处理Execel表格 , 当然不一定非要用Java 我们在这里只说在Java里怎么去处理表格 . Easy Excel 我们今天要说到就是 Easy Excel , 这个是阿里的项…

学习笔记整理-JS-01-语法与变量

文章目录 一、语法与变量1. 初识JavaScript2. JavaScript的历史3. JavaScript与ECMAScript的关系4. JavaScript的体系5. JavaScript的语言风格和特性 二、语法1. JavaScript的书写位置2. 认识输出语句3. REPL环境&#xff0c;交互式解析器4. 变量是什么5. 重点内容 一、语法与变…

无涯教程-Perl - redo函数

描述 此函数将重新启动当前循环,而不会强制判断控制语句。块中不再执行任何语句。如果存在继续块,将不会执行。如果指定了LABEL,则在LABEL标识的循环开始时重新开始执行。 语法 以下是此函数的简单语法- redo LABELredo返回值 此函数不返回任何值。 例 以下是显示其基本…

J35复杂链表的复制

题目地址&#xff1a;复杂链表的复制_牛客题霸_牛客网 题目回顾&#xff1a; 解题思路&#xff1a; 这里我们采用双指针的方法。 我们在创建节点的时候&#xff0c;可能当前节点创建了&#xff0c;但是当前节点的随机指针指向的节点还没创建&#xff0c;这种情况下&#xff…

【python】-【】

文章目录 转义字符和原字符二进制与字符编码标识符和保留字变量的定义和使用变量字符串列表for 一、print会输出①数字②字符串&#xff08;必须加引号&#xff09;③含有运算符的表达式&#xff08;例如 31 其中3&#xff0c;1是操作数&#xff0c;是运算符&#xff09;&#…

zabbix监控mysql数据库、nginx、Tomcat

文章目录 一.zabbix监控mysql数据库1.环境规划2.zabbix-server安装部署&#xff08;192.168.198.17&#xff09;3.zabbix-mysql安装部署&#xff08;192.168.198.15&#xff09;3.1 部署 zabbix 客户端3.2 服务端验证 zabbix-agent2 的连通性&#xff08;192.168.198.17&#x…

使用script标签解决跨域问题,但是只能使用get请求,且不需要获取get请求的数据,例如埋点,只需要触发后发送get请求,而不需要获取返回的参数

在项目中&#xff0c;使用埋点的时候&#xff0c;因为使用的是外部提供的接口&#xff0c;所以直接请求的时候&#xff0c;前端会报跨域的问题&#xff0c;本着不麻烦后端的想法&#xff0c;怎怎么前端实现跨域而完全不需要后段的配合&#xff0c;这时候就想到了通过script标签…

Steam搬砖项目

最近分享了不少关于Steam搬砖项目的文章&#xff0c;但是也有一部分人说&#xff0c;项目不知道如何赚钱的。 今天这篇文章&#xff0c;就分享一下&#xff0c;对于Steam游戏搬砖项目&#xff0c;他是靠什么来挣钱的。 那讲解之前&#xff0c;先回复一下&#xff0c;另外一个问…

【rust/egui】(二)看看template的main函数:日志输出以及eframe run_native

说在前面 rust新手&#xff0c;egui没啥找到啥教程&#xff0c;这里自己记录下学习过程环境&#xff1a;windows11 22H2rust版本&#xff1a;rustc 1.71.1egui版本&#xff1a;0.22.0eframe版本&#xff1a;0.22.0上一篇&#xff1a;这里 开始 首先让我们看看main.rs中有些什么…

纷享销客张睿:每年见100家快消企业,我发现数字化的价值本质

“ 快消行业的竞争从马拉松进入八角笼中&#xff0c;重度内卷将成行业面临的普遍挑战&#xff0c;”新经销创始人赵波在2023年第四届中国创新食品大会感叹。 8月4-6日&#xff0c;2023年第四届中国创新食品大会暨粤港澳大湾区食品博览会在广东东莞举行&#xff0c;汇聚嘉士利、…

【解读Spikingjelly】使用单层全连接SNN识别MNIST

原文档&#xff1a;使用单层全连接SNN识别MNIST — spikingjelly alpha 文档 代码地址&#xff1a;完整的代码位于activation_based.examples.lif_fc_mnist.py GitHub - fangwei123456/spikingjelly: SpikingJelly is an open-source deep learning framework for Spiking Neur…

解决WSL2的docker删除镜像后,磁盘空间不释放问题

1、问题原因 由于WSL2本质上是虚拟机&#xff0c;所以 Windows 会自动创建 vhdx 后缀的虚拟磁盘文件作为存储。这个 vhdx 后缀的虚拟磁盘文件特点是可以自动扩容&#xff0c;但是一般不会自动缩容。一旦有很多文件把它“撑大”&#xff0c;即使把这些文件删除它也不会自动“缩…

go内存管理机制

golang内存管理基本是参考tcmalloc来进行的。go内存管理本质上是一个内存池&#xff0c;只不过内部做了很多优化&#xff1a;自动伸缩内存池大小&#xff0c;合理切割内存块。 基本概念&#xff1a; Page&#xff1a;页&#xff0c;一块 8 K大小的内存空间。Go向操作系统申请和…

Rabbitmq消息不丢失

目录 一、消息不丢失1.消息确认2.消息确认业务封装2.1 发送确认消息测试2.2 消息发送失败&#xff0c;设置重发机制 一、消息不丢失 消息的不丢失&#xff0c;在MQ角度考虑&#xff0c;一般有三种途径&#xff1a; 1&#xff0c;生产者不丢数据 2&#xff0c;MQ服务器不丢数据…

MySQL入门学习教程(二)

上一篇文章讲的是mysql的基本操作&#xff0c;这一篇会有一点难以理解&#xff0c;本节主要内容mysql视图&#xff0c;存储过程&#xff0c;函数&#xff0c;事务&#xff0c;触发器&#xff0c;以及动态执行sql 视图view 视图是一个虚拟表&#xff0c;其内容由查询定义。同真…

在Java中操作Redis(详细-->从环境配置到代码实现)

在Java中操作Redis 文章目录 在Java中操作Redis1、介绍2、Jedis3、Spring Data Redis3.1、对String的操作3.2、对哈希类型数据的操作3.3、对list的操作3.4、对set类型的操作3.5、对 ZSet类型的数据&#xff08;有序集合&#xff09;3.6、通用类型的操作 1、介绍 Redis 的Java客…

开发者必知:.gitignore 文件的魔法,助你管理项目文件如虎添翼!

前言&#xff1a; 在软件开发的世界中&#xff0c;版本控制是一个至关重要的环节。而 Git 作为目前最流行的分布式版本控制系统之一&#xff0c;已经成为开发者不可或缺的工具。然而&#xff0c;在日常的开发过程中&#xff0c;有些文件是不适合被纳入版本控制的&#xff0c;比…

【C++入门】const 成员函数

文章目录 一、基本概念二、经典问题三、使用建议 一、基本概念 const 修饰的成员函数就称作 const 成员函数。 例子&#xff1a; class Date { public:void Display() const{...}private:int _year;int _month;int _day; };事实上&#xff0c;const 成员函数的这个 const 修…

Linux文件系统管理

Linux文件系统管理 磁盘的组成与分区 计算机用于存取文件的硬件是磁盘&#xff0c;磁盘的组成主要有磁盘盘、机械手臂、磁盘读取头与主轴马达所组成&#xff0c; 而数据的写入其实是在磁盘盘上面。磁盘盘上面又可细分出扇区(Sector)与磁道(Track)两种单位&#xff0c; 其中扇区…