【优选算法-滑动窗口】长度最小的子数组、无重复字符的最长子串、最大连续1的个数、将x减为0的最小操作数、水果成篮

一、长度最小的子数组

题目链接:

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

题目介绍:

给定一个含有 n 个正整数的数组和一个正整数 target 。

找出该数组中满足其总和大于等于 target 的长度最小的 子数组

 [numsl, numsl+1, ..., numsr-1, numsr] ,并返回其长度如果不存在符合条件的子数组,返回 0 。

思路1:暴力枚举

每次固定一个起始下标,往后遍历求和,直到大于target停止,并更新最小长度

class Solution {
public:
    int minSubArrayLen(int target, vector<int>& nums) {
        // 记录结果
        int ret = INT_MAX;
        int n = nums.size();
        // 枚举出所有满⾜和⼤于等于 target 的⼦数组[start, end]
        // 由于是取到最⼩,因此枚举的过程中要尽量让数组的⻓度最⼩
        // 枚举开始位置
        for (int start = 0; start < n; start++) {
            int sum = 0; // 记录从这个位置开始的连续数组的和
            // 寻找结束位置
            for (int end = start; end < n; end++) {
                sum += nums[end]; // 将当前位置加上

                if (sum >= target) // 当这段区间内的和满⾜条件时
                {
                    // 更新结果,start 开头的最短区间已经找到
                    ret = min(ret, end - start + 1);
                    break;
                }
            }
        }
        // 返回最后结果
        return ret == INT_MAX ? 0 : ret;
    }
};
思路2:对思路1进行优化(滑动窗口)

        思路1在从一个下标开始,计算完后,sum会重新归为0,从下一个下标开始重新计算求和,但是这里有一个问题,第一次求和与第二次求和只相差了一个元素的值,如果直接从头开始,就会产生重重复的计算,所以每当我们从一个下标计算完最小长度后,可以利用单调性,不让sum归0,也不改变end了,直接让sum -= nums[start] ,然后让start++,这样就可以继续向后计算了,但是要注意的是,最左边的值可能是一个很小的值,甚至,sum减去它以后还是满足大于target的,所以这里不能用if判断,而是用一个while循环。

          这样在一个连续空间上,两个指针同向移动就叫做滑动窗口

时间复杂度:虽然代码是两层循环,但是我们的 left 指针和 right 指针都是不回退的,两者最多都往后移动 n 次。因此时间复杂度是 O(N) 。

class Solution {
public:
    int minSubArrayLen(int target, vector<int>& nums) {
        int  minlen=INT_MAX;
        int sum=0;
        for(int left=0,right=0;right<nums.size();right++)
        {
            sum+=nums[right];
            while(sum>=target)
            {
                minlen=min(minlen,right-left+1);
                sum-=nums[left];
                left++;
            }
        }
        return minlen==INT_MAX?0:minlen;;
    }
};

二、无重复字符的最长子串

题目链接:

3. 无重复字符的最长子串 - 力扣(LeetCode)

题目介绍:

给定一个字符串 s ,请你找出其中不含有重复字符的 最长子串 的长度。

  • 0 <= s.length <= 5 * 104
  • s 由英文字母、数字、符号和空格组成
思路1:暴力枚举

依旧是从一个下标开始遍历,判断是否出现重复字符串可以利用哈希表,当hash[n]>1时就说明有重复字符了

class Solution {
public:
    int lengthOfLongestSubstring(string s) {
        int ret = 0; // 记录结果
        int n = s.length();
        // 1. 枚举从不同位置开始的最⻓重复⼦串
        // 枚举起始位置
        for (int i = 0; i < n; i++) {
            // 创建⼀个哈希表,统计频次
            int hash[128] = {0};

            // 寻找结束为⽌
            for (int j = i; j < n; j++) {
                hash[s[j]]++;       // 统计字符出现的频次
                if (hash[s[j]] > 1) // 如果出现重复的
                    break;

                // 如果没有重复,就更新 ret
                ret = max(ret, j - i + 1);
            }
        }
        // 2. 返回结果
        return ret;
    }
};
思路2:滑动窗口

研究的对象依旧是⼀段连续的区间,因此继续使用「滑动窗口」思想来优化。

让滑动窗口满足:窗口内所有元素都是不重复的。 做法:

右端元素 ch 进⼊窗口的时候,哈希表统计这个字符的频次:

  • 如果这个字符出现的频次超过 1 ,说明窗口内有重复元素,那么就从左侧开始划出窗口, 直到 ch 这个元素的频次变为 1 ,然后再更新结果。
  • 如果没有超过1 ,说明当前窗口没有重复元素,可以直接更新结果
class Solution {
public:
    int lengthOfLongestSubstring(string s) {
        int hash[128] = {0};
        int maxlen = 0;
        for (int left = 0, right = 0; right < s.size(); right++) {
            hash[s[right]]++;
            while (hash[s[right]] > 1) {
                hash[s[left]]--;
                left++;
            }
            maxlen = max(maxlen, right - left + 1);
        }
        return maxlen;
    }
};

三、最大连续 1 的个数 III

题目链接:

1004. 最大连续1的个数 III - 力扣(LeetCode)

题目介绍:

给定一个二进制数组 nums 和一个整数 k,假设最多可以翻转 k 个 0 ,则返回执行操作后 数组中连续 1 的最大个数 。

思路:

不用想着翻转元素,这样反而会把题目搞复杂,转变一下思路,这个题目就是找一个0的个数不大于k个的连续区间的最大长度,同样的在一段连续区间上操作,可以采用滑动窗口。

定义一个变量num用于记录窗口中0的个数,每次让nums[right]入窗口,如果他的值是0的话,就让num++,如果num>k,就从左边出窗口,直到 num <= k,再更新结果;如果入窗口后,num还是小于k的,说明这段区间是符合要求的,可以直接更新结果

class Solution {
public:
    int longestOnes(vector<int>& nums, int k) {
        int num=0;
        int maxlen=INT_MIN;
        for(int left=0,right=0;right<nums.size();right++)
        {
            if(nums[right]==0)
                num++;
            while(num>k)
            {
                if(nums[left]==0)
                {
                    left++;
                    num--;
                }
                else
                {
                    left++;
                }
            }
            maxlen=max(maxlen,right-left+1);
        }
        return maxlen;
    }
};

四、将 x 减到 0 的最小操作数

题目链接:

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

题目介绍:

给你一个整数数组 nums 和一个整数 x 。每一次操作时,你应当移除数组 nums 最左边或最右边的元素,然后从 x 中减去该元素的值。请注意,需要 修改 数组以供接下来的操作使用。

如果可以将 x 恰好 减到 0 ,返回 最小操作数 ;否则,返回 -1 。

  • 1 <= nums.length <= 105
  • 1 <= nums[i] <= 104
  • 1 <= x <= 109
思路:

这个题目如果从正面想的话比较复杂,因为它是从数组两边进行操作的,所以我们可以转变一下思路,将这个题变为找一段最长连续区间的和=sum - x,这样最小操作数就是数组的长度减去这个连续区间的长度了。

首先计算出数组的和,然后求出sum-x,如果target的值都小于0了,就返回1,因为数组中的元素都是大于0的。

接下来就可以让right入窗口了,如果和大于target的话,就出窗口,直到和小于等于target,要注意更新结果要有一个判断条件,就是 sum=target

class Solution {
public:
    int minOperations(vector<int>& nums, int x) {
        //转变思路,求一段连续空间中和为sum-x的长度
        int sum=0;
        for(auto& e: nums)
        {
            sum+=e;
        }
        int target=sum-x;
        if(target<0)
        return -1;
        int maxlen=-1;
        sum=0;
        for(int left=0,right=0;right<nums.size();right++)
        {
            sum+=nums[right];
            
            while(sum>target)
            {
                sum-=nums[left++];
            }
            if(sum==target)
            maxlen=max(maxlen,right-left+1);
        }
        if(maxlen==-1)
        return -1;
        else
        return nums.size()-maxlen;
    }
};

五、水果成篮

题目链接:

904. 水果成篮 - 力扣(LeetCode)

题目介绍:

你正在探访一家农场,农场从左到右种植了一排果树。这些树用一个整数数组 fruits 表示,其中 fruits[i] 是第 i 棵树上的水果 种类 。

你想要尽可能多地收集水果。然而,农场的主人设定了一些严格的规矩,你必须按照要求采摘水果:

  • 你只有 两个 篮子,并且每个篮子只能装 单一类型 的水果。每个篮子能够装的水果总量没有限制。
  • 你可以选择任意一棵树开始采摘,你必须从 每棵 树(包括开始采摘的树)上 恰好摘一个水果 。采摘的水果应当符合篮子中的水果类型。每采摘一次,你将会向右移动到下一棵树,并继续采摘。
  • 一旦你走到某棵树前,但水果不符合篮子的水果类型,那么就必须停止采摘。

给你一个整数数组 fruits ,返回你可以收集的水果的 最大 数目。

思路:

这个题目依旧是操作一个连续的空间,也要遍历,所以可以采用滑动窗口,让滑动窗口维护只有两类水果。

当一个水果入窗口后,可以将他的种类用哈希表维护起来,这样当一个新水果进来后,哈希表的长度就会加1,如果哈西表的长度大于2的话,说明窗口中存在第三种水果了,那就从左边出窗口,直到哈希表的长度小于2

class Solution {
public:
    int totalFruit(vector<int>& fruits) {
        unordered_map<int,int> m;
        int num=0;
        for(int left=0,right=0;right<fruits.size();right++)
        {
           m[fruits[right]]++;

           while(m.size()>2)
           {
                m[fruits[left]]--;
                if(m[fruits[left]]==0)
                m.erase(fruits[left]);
                left++;
           }
           num=max(num,right-left+1);
        }
        return num;
    }
};

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

appium学习之二:adb命令

1、查看设备 adb devices 2、连接 adb connect IP:端口 3、安装 adb install xxx.apk 4、卸载 adb uninstall 【包名】 5、把对应目录下的1.txt文件传到手机sdcard下 adb push 1.txt /sdcard 6、进入对应的设备里 adb shell 7、切入sdcard目录 cd /sdcard 8、ls 查…

算法——差分

差分可以看作是前缀和的逆运算&#xff0c;前缀和可以帮我们快速得到某个区间的和&#xff0c;而差分就是我们将原数组看作是一个前缀和数组&#xff08;q[]&#xff09;我们去构造一个差分数组&#xff08;c[]&#xff09; 一维差分 使存在如下关系&#xff1a; q[i] c[1]…

使用 EasyExcel 提升 Excel 处理效率

目录 前言1. EasyExcel 的优点2. EasyExcel 的功能3. 在项目中使用 EasyExcel3.1 引入依赖3.2 实体类的定义与注解3.3 工具类方法的实现3.4 在 Controller 中使用 4. 总结5. 参考地址 前言 在日常开发中&#xff0c;Excel 文件的处理是不可避免的一项任务&#xff0c;特别是在…

健康管理系统(Koa+Vue3)

系统界面(源码末尾获取) 系统技术 Vue3 Koa Nodejs Html Css Js ....... 系统介绍 系统比较简单,轻轻松松面对结业课堂作业.采用的是基于nodejs开发的Koa框架作为后端,采用Vue框架作为前端,完成快速开发和界面展示. 系统获取 啊啊啊宝/KoaVue3https://gitee.com/ah-ah-b…

python进阶-05-利用Selenium来实现动态爬虫

python进阶-05-利用Selenium来实现动态爬虫 一.说明 这是python进阶部分05&#xff0c;我们上一篇文章学习了Scrapy来爬取网站&#xff0c;但是很多网站需要登录才能爬取有用的信息&#xff0c;或者网站的静态部分是一个空壳&#xff0c;内容是js动态加载的,或者人机验证&…

day10性能测试(2)——Jmeter

【没有所谓的运气&#x1f36c;&#xff0c;只有绝对的努力✊】 目录 1、LoadRunner vs Jmeter 1.1 LoadRunner 1.2 Jmeter 1.3 对比小结 2、Jmeter 环境安装 2.1 安装jdk 2.2 安装Jmeter 2.3 小结 3、Jmeter 文件目录结构 4、Jmeter默认配置修改 5、Jmeter元件、组…

架构15-服务网格

零、文章目录 架构15-服务网格 1、透明通信的涅槃 &#xff08;1&#xff09;服务网格 概念 服务网格是一种处理程序间通信的基础设施&#xff0c;主要由数据平面和控制平面组成。它通过边车代理和控制程序管理程序间的通信&#xff0c;弥补了容器编排系统对分布式应用细粒…

day08 接口测试(4)知识点完结!!

【没有所谓的运气&#x1f36c;&#xff0c;只有绝对的努力✊】 目录 1、postman读取外部数据文件&#xff08;参数化&#xff09; 1.1 数据文件简介 1.2 导入外部数据文件 1.2.1 csv文件 1.2.2 导入 json文件 1.3 读取数据文件数据 1.4 案例 1.5 生成测试报告 2、小…

2024年11月HarmonyOS应用开发者高级认证全新题库

注意事项&#xff1a;切记在考试之外的设备上打开题库进行搜索&#xff0c;防止切屏三次考试自动结束&#xff0c;题目是乱序&#xff0c;每次考试&#xff0c;选项的顺序都不同&#xff0c;作者已于2024年11月22日又更新了一波题库&#xff0c;题库正确率99%&#xff01; 新版…

【Python网络爬虫 常见问题汇总】

目录 1. 爬取图片出现403解决办法&#xff1a;设置请求头中的Referer字段 2.关于干坏事的问题后续不定期更新 欢迎共同探讨学习进步 1. 爬取图片出现403 问题出自案例9&#xff0c;已解决。 【Python网络爬虫笔记】9- 抓取优美图库高清壁纸 当在爬取图库图片时遇到 403 错误…

《探索形象克隆:科技与未来的奇妙融合》

目录 一、什么是形象克隆 二、形象克隆的技术原理 三、形象克隆的发展现状 四、形象克隆的未来趋势 五、形象克隆的应用场景 六、形象克隆简单代码案例 Python 实现数字人形象克隆 Scratch 实现角色克隆效果&#xff08;以猫为例&#xff09; JavaScript 实现 Scratc…

Mac软件推荐

Mac软件推荐 截图SnipasteXnipBob 快捷启动Raycast 系统检测Stats 解压缩The UnarchiverKeka&#xff08;付费&#xff09; 视频播放IINA 视频下载Downie&#xff08;付费&#xff09; 屏幕刘海TopNotchMediaMate&#xff08;付费&#xff09;NotchDrop&#xff08;付费&#x…

Linux——linux系统移植

创建VSCode工程 1、将NXP官方的linux内核拷贝到Ubuntu 2、解压缩tar -vxjf linux-imx-rel_imx_4.1.15_2.1.0_ga.tar.bz2 NXP官方开发板Linux内核编译 1、将.vscode文件夹复制到NXP官网linux工程中&#xff0c;屏蔽一些不需要的文件 2、编译NXP官方EVK开发板对应的Linux系统…

TikTok运营选什么网络?要用原生IP吗?

不管是跨境电商运营还是个人IP打造&#xff0c;TikTok都是必不可少的一个大流量平台。但运营TikTok必然要面临网络问题&#xff0c;如果没有妥当的解决方案&#xff0c;不仅难以获取流量&#xff0c;还可能面临封号的风险。因此&#xff0c;选择可靠的网络并合理使用是非常重要…

Spring 基础

什么是 Spring 框架? Spring 是一款开源的轻量级 Java 开发框架&#xff0c;旨在提高开发人员的开发效率以及系统的可维护性。 我们一般说 Spring 框架指的都是 Spring Framework&#xff0c;它是很多模块的集合&#xff0c;使用这些模块可以很方便地协助我们进行开发&#…

Redis篇-5--原理篇4--Lua脚本

1、概述 Redis 支持使用 Lua 脚本来执行复杂的操作&#xff0c;这为 Redis 提供了更强的灵活性和性能优化能力。通过 Lua 脚本&#xff0c;你可以在服务器端执行一系列命令&#xff0c;而不需要多次往返客户端与服务器之间&#xff0c;从而减少了网络延迟并提高了效率。此外&a…

【数据库】关系代数和SQL语句

一 对于教学数据库的三个基本表 学生S(S#,SNAME,AGE,SEX) 学习SC(S#,C#,GRADE) 课程(C#,CNAME,TEACHER) &#xff08;1&#xff09;试用关系代数表达式和SQL语句表示&#xff1a;检索WANG同学不学的课程号 select C# from C where C# not in(select C# from SCwhere S# in…

【git】--- 通过 git 和 gitolite 管理单仓库的 SDK

在编程的艺术世界里,代码和灵感需要寻找到最佳的交融点,才能打造出令人为之惊叹的作品。而在这座秋知叶i博客的殿堂里,我们将共同追寻这种完美结合,为未来的世界留下属于我们的独特印记。【git】--- 通过 git 和 gitolite 管理单仓库的 SDK 开发环境一、安装配置 gitolite二…

HDFS高可用模式安装部署

实验步骤 将ZooKeeper集群模式启动获取安装包 安装包在本地&#xff1a;通过XFTP等工具将安装包上传到虚拟机中安装包在网络&#xff1a; 虚拟机可以访问互联网虚拟机无法访问互联网解压缩安装包将解压出来安装目录重命名配置环境变量刷新环境变量&#xff0c;使新增的环境变量…

mysql程序介绍,选项介绍(常用选项,指定选项的方式,特性),命令介绍(查看,部分命令),从sql文件执行sql语句的两种方法

目录 mysql程序 介绍 选项 介绍 常用选项 指定选项的方式 ​编辑配置文件 环境变量 选项特性 指定选项 选项名 选项值 命令 介绍 查看客户端命令 tee/notee prompt source system help contents 从.sql文件执行sql语句 介绍 方式 source 从外部直接导入…