滑动窗口习题篇(下)

devtools/2024/11/7 14:54:15/

滑动窗口习题篇(下)

1.leetcode.cn/problems/fruit-into-baskets" rel="nofollow">水果成篮

题目描述:

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

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

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

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

示例 1:

​ 输入:fruits = [1,2,1]

​ 输出:3

解释:

​ 可以采摘全部 3 棵树。

示例 2:

​ 输入:fruits = [0,1,2,2]

​ 输出:3

解释:

​ 可以采摘 [1,2,2] 这三棵树。

​ 如果从第⼀棵树开始采摘,则只能采摘 [0,1] 这两棵树。

示例 3:

​ 输入:fruits = [3,3,3,1,2,1,1,2,3,3,4]

​ 输出:5

解释:

​ 可以采摘 [1,2,1,1,2] 这五棵树。

问题转化:找出一个最长的子数组的长度,子数组中不超过两种类型的水果。

解法一:暴力枚举+哈希表

优化:从前往后进行枚举,即left++时,这时水果的种类不变,right不变;水果的种类变小,right右移。

解法二:滑动窗口

算法思路:

研究的对象是一段连续的区间,可以使用滑动窗口思想来解决问题。

1.进窗口判断条件:

窗口内水果的种类只有两种.

滑动窗口做法:

1.右端水果进入窗口的时候,用哈希表统计这个水果的频次(进窗口)。

2.这个水果进来后,判断哈希表的大小

  • 如果大小超过 2:说明窗口内水果种类超过了两种。那么就从左侧开始依次将水果划出窗口(出窗口),直到哈希表的大小小于等于 2,然后更新结果
  • 如果没有超过 2,说明当前窗口内水果的种类不超过两种,直接更新结果 ret
算法流程:
  1. 初始化哈希表 hash 来统计窗口内水果的种类和数量;

  2. left = 0,right = 0,记结果变量 ret = 0;

  3. right < f.size() ,下列一直循环:

  • 将当前水果放入哈希表中(进窗口);

  • 判断当前水果进来后,哈希表的大小:

    • 如果超过 2:(出窗口
    • 将左侧元素滑出窗口,并且在哈希表中将该元素的频次减一;
    • 如果这个元素的频次减一之后变成了 0,就把该元素从哈希表中删除;
    • 重复上述两个过程,直到哈希表中的大小不超过 2;
  • 更新结果 ret;

  • right++,让下⼀个元素进入窗口;

  1. 循环结束后,ret 存的就是最终结果。
代码实现(使用容器):
class Solution
{
public:int totalFruit(vector<int>& f) {unordered_map<int, int> hash; // 统计窗⼝内出现了多少种⽔果int ret = 0;for(int left = 0, right = 0; right < f.size(); right++){hash[f[right]]++; // 进窗⼝while(hash.size() > 2) // 判断{// 出窗⼝hash[f[left]]--;if(hash[f[left]] == 0)hash.erase(f[left]);left++;}ret = max(ret, right - left + 1);}return ret;}
};
实现(用数组模拟哈希表):
class Solution
{
public:int totalFruit(vector<int>& f) {int hash[100001] = { 0 }; // 统计窗⼝内出现了多少种⽔果int ret = 0;for(int left = 0, right = 0, kinds = 0; right < f.size(); right++){if(hash[f[right]] == 0) kinds++; // 维护⽔果的种类hash[f[right]]++; // 进窗⼝while(kinds > 2) // 判断{// 出窗⼝hash[f[left]]--;if(hash[f[left]] == 0) kinds--;left++;}ret = max(ret, right - left + 1);}return ret;}
};

2.leetcode.cn/problems/find-all-anagrams-in-a-string" rel="nofollow">找到字符串中所有字母异位词

题目描述:

给定两个字符串 sp,找到 s 中所有 p异位词的子串,返回这些子串的起始索引。不考虑答案输出的顺序。

异位词指由相同字母重排列形成的字符串(包括相同的字符串)。

示例 1:

​ 输入: s = “cbaebabacd”, p = “abc
​ 输出: [0,6]
解释:
​ 起始索引等于 0 的子串是 “cba”, 它是 “abc” 的异位词。
​ 起始索引等于 6 的子串是 “bac”, 它是 “abc” 的异位词。

示例 2:

​ 输入: s = “abab”, p = “ab
​ 输出: [0,1,2]
解释:
​ 起始索引等于 0 的子串是 “ab”, 它是 “ab” 的异位词。
​ 起始索引等于 1 的子串是 “ba”, 它是 “ab” 的异位词。
​ 起始索引等于 2 的子串是 “ab”, 它是 “ab” 的异位词。

提示:

  • 1 <= s.length, p.length <= 3 * 104
  • sp 仅包含小写字母

算法思路:

本题研究对象连续的区间,我们使用滑动窗口思想来解决这道题。

1.如何快速判断两个字符串是否是“异位词”

因为字符串 p 的异位词的长度⼀定与字符串 p 的长度相同,所以我们可以在字符串 s 中构造一个长度为与字符串 p 的长度相同的滑动窗口,并在滑动中维护窗口中每种字母的数量;

  • 当窗口中每种字母的数量与字符串 p 中每种字母的数量相同时,则说明当前窗口为字符串 p的异位词;

  • 因此可以用两个大小为 26 的数组来模拟哈希表。用hash1来保存 p 中每⼀个字符出现的个数,hash2来保存 s 中的子串每个字符出现的个数。这样就能判断两个串是否是异位词。

2.进窗口判断条件:

right-left+1>m(字符串 p 的异位词的长度).

滑动窗口做法:

  1. hash1统计字符串 p中每个字符出现的个数,hash2统计窗口里面的每⼀个字符出现的个数;

  2. left=0,right=0,count=0(用来统计窗口中有效字符的个数);

  3. right<s.size() ,下列一直循环:

  • 如果right-left+1<=m时,将右端元素划入窗口,当hash2[in]<=hash1[in]时,count++;

  • 如果right-left+1>m时,将左端元素划出窗口,这时要是hash2[out]<=hash1[out]时,count- -;

  1. 更新结果:当count==m时,返回ret的值。

代码实现:

class Solution
{
public:vector<int> findAnagrams(string s, string p) {vector<int> ret;int hash1[26] = { 0 }; // 统计字符串 p 中每个字符出现的个数for(auto ch : p) hash1[ch - 'a']++;int hash2[26] = { 0 }; // 统计窗⼝⾥⾯的每⼀个字符出现的个数int m = p.size();for(int left = 0, right = 0, count = 0; right < s.size(); right++){char in = s[right];// 进窗⼝ + 维护 countif(++hash2[in - 'a'] <= hash1[in - 'a']) count++; if(right - left + 1 > m) // 判断{char out = s[left++];// 出窗⼝ + 维护 countif(hash2[out - 'a']-- <= hash1[out - 'a']) count--; }// 更新结果if(count == m) ret.push_back(left);}return ret;}
};

3.leetcode.cn/problems/substring-with-concatenation-of-all-words" rel="nofollow">串联所有单词的子串

题目描述:

给定一个字符串 s 和一个字符串数组 words words 中所有字符串 长度相同

s 中的 串联子串 是指一个包含 words 中所有字符串以任意顺序排列连接起来的子串。

  • 例如,如果 words = ["ab","cd","ef"], 那么 "abcdef""abefcd""cdabef""cdefab""efabcd", 和 "efcdab" 都是串联子串。 "acdbef" 不是串联子串,因为他不是任何 words 排列的连接。

返回所有串联子串在 s 中的开始索引。你可以以 任意顺序 返回答案。

示例 1:

​ 输入:s = "barfoothefoobarman", words =["foo","bar"]
​ 输出:[0,9]
解释:

因为words.length == 2同时 words[i].length == 3,连接的子字符串的长度必须为 6。
子串"barfoo"开始位置是 0。它是 words 中以 [“bar”,“foo”] 顺序排列的连接。
子串 "foobar" 开始位置是 9。它是 words 中以 [“foo”,“bar”] 顺序排列的连接。
输出顺序无关紧要。返回 [9,0] 也是可以的。

示例 2:

​ 输入:s = “wordgoodgoodgoodbestword”, words = ["word","good","best","word"]
​ 输出:[]
解释:

因为 words.length == 4 并且 words[i].length == 4,所以串联子串的长度必须为 16。
s 中没有子串长度为 16 并且等于 words 的任何顺序排列的连接。
所以我们返回一个空数组。

示例 3:

​ 输入:s = "barfoofoobarthefoobarman", words = ["bar","foo","the"]
​ 输出:[6,9,12]
解释:

因为 words.length == 3 并且 words[i].length == 3,所以串联子串的长度必须为 9。
子串 "foobarthe" 开始位置是 6。它是 words 中以 ["foo","bar","the"] 顺序排列的连接。
子串 "barthefoo" 开始位置是 9。它是 words 中以 ["bar","the","foo"] 顺序排列的连接。
子串 "thefoobar" 开始位置是 12。它是 words 中以 ["the","foo","bar"] 顺序排列的连接。

提示:

  • 1 <= s.length <= 104
  • 1 <= words.length <= 5000
  • 1 <= words[i].length <= 30
  • words[i]s 由小写英文字母组成

算法思路:

解法:滑动窗口+哈希表

如果我们把每一个单词看成一个一个字母,问题就变成了上面第二题——找到字符串中所有的字母异位词」。

不同在于之前处理的对象是一个一个的字符,我们这里处理的对象是一个一个的单词。

代码实现:

class Solution
{
public:vector<int> findSubstring(string s, vector<string>& words) {vector<int> ret;unordered_map<string, int> hash1; // 保存 words ⾥⾯所有单词的频次for(auto& s : words) hash1[s]++;int len = words[0].size(), m = words.size();for(int i = 0; i < len; i++) // 执⾏ len 次{unordered_map<string, int> hash2; // 维护窗⼝内单词的频次for(int left = i, right = i, count = 0; right + len <= s.size(); right += len){// 进窗⼝ + 维护 countstring in = s.substr(right, len);hash2[in]++;if(hash1.count(in) && hash2[in] <= hash1[in]) count++;// 判断if(right - left + 1 > len * m){// 出窗⼝ + 维护 countstring out = s.substr(left, len);if(hash1.count(out) && hash2[out] <= hash1[out]) count--;hash2[out]--;left += len;}// 更新结果if(count == m) ret.push_back(left);}}return ret;}
};

4.leetcode.cn/problems/minimum-window-substring" rel="nofollow">最小覆盖子串

题目描述:

给你一个字符串 s 、一个字符串 t 。返回 s 中涵盖 t 所有字符的最小子串。如果 s 中不存在涵盖 t 所有字符的子串,则返回空字符串 ""

注意:

  • 对于 t 中重复字符,我们寻找的子字符串中该字符数量必须不少于 t 中该字符数量。
  • 如果 s 中存在这样的子串,我们保证它是唯一的答案。

示例 1:

输入:s = "ADOBECODEBANC", t = "ABC"
输出:"BANC"
解释:最小覆盖子串 “BANC” 包含来自字符串 t 的 ‘A’、‘B’ 和 ‘C’。

示例 2:

输入:s = "a", t = "a"
输出:"a"
解释:整个字符串 s 是最小覆盖子串。

示例 3:

输入: s = “a”, t = “aa
输出: “”
解释: t 中两个字符 ‘a’ 均应包含在 s 的子串中,
因此没有符合条件的子字符串,返回空字符串。

提示:

  • m == s.length
  • n == t.length
  • 1 <= m, n <= 105
  • st 由英文字母组成

解法一:暴力枚举+哈希表

解法二:滑动窗口+哈希表

算法思路:

研究对象是连续的区间,因此可以尝试使用滑动窗口的思想来解决。

1.如何判断当前窗口内的所有字符是符合要求的呢?

我们可以使用两个哈希表,其中⼀个将目标串的信息统计起来,另一个哈希表动态的维护窗口内字符串的信息。

当动态哈希表中包含⽬标串中所有的字符,并且对应的个数都不小于目标串的哈希表中各个字符的个数,那么当前的窗口就是⼀种可行的方案。

算法流程:

1.进窗口判断条件:

hash2中的有效字符大于hash1中的有效字符.

滑动窗口做法:

  1. hash1用来统计字符串 t 中每⼀个字符的频次,hash2用来统计窗⼝内每个字符的频次;
  2. left=0,right=0,count=0;
  3. right<s.size() ,下列一直循环:
  • 如果right-left+1<=minlen时,将右端元素划入窗口,当hash2[in]==hash1[in]时,count++;

  • 如果right-left+1>minlen时,将左端元素划出窗口,这时要是hash2[out]==hash1[out]时,count- -;

  1. 更新结果:起始位置和最短长度

优化:判断条件:

使用变量count标记有效字符的种类

1.进窗口:当hash2[in]==hash1[in],count++;

2.出窗口:当hash2[out]==hash1[out],count- -;

3.判断条件:count==hash1.size()

代码实现:
class Solution
{
public:string minWindow(string s, string t) {int hash1[128] = { 0 }; // 统计字符串 t 中每⼀个字符的频次int kinds = 0; // 统计有效字符有多少种for(auto ch : t)if(hash1[ch]++ == 0) kinds++;int hash2[128] = { 0 }; // 统计窗⼝内每个字符的频次int minlen = INT_MAX, begin = -1;for(int left = 0, right = 0, count = 0; right < s.size(); right++){char in = s[right];if(++hash2[in] == hash1[in]) count++; // 进窗⼝ + 维护 countwhile(count == kinds) // 判断条件{if(right - left + 1 < minlen) // 更新结果{minlen = right - left + 1;begin = left;}char out = s[left++];if(hash2[out]-- == hash1[out]) count--; // 出窗⼝ + 维护 count}}if(begin == -1) return "";else return s.substr(begin, minlen);}
};

最后,本篇文章到此结束,感觉不错的友友们可以一键三连支持一下笔者,有任何问题欢迎在评论区留言哦~


http://www.ppmy.cn/devtools/131710.html

相关文章

Java中的远程方法调用——RPC详解

1. 什么是RPC&#xff1f; RPC基础介绍 Java中的远程方法调用&#xff08;Remote Procedure Call&#xff0c;RPC&#xff09;是一种允许一个程序调用另一台计算机上方法的技术&#xff0c;就像在本地一样。RPC的核心思想是简化分布式计算&#xff0c;让我们可以跨网络调用远程…

智谱发布AI助理,帮人类敲响AGI的大门

人工智能之父John McCarthy曾说&#xff1a;“只要AI可以开始正常工作&#xff0c;就不会有人再把它当AI了。”如今&#xff0c;这一预言正在逐渐变为现实。 10月25日&#xff0c;智谱AI推出了自主智能体AutoGLM&#xff0c;能够模拟人类操作手机&#xff0c;执行各种任务。 …

Vuex的基本使用

文章目录 一、Vuex概述1.是什么2.使用场景3.优势4.注意二、如何构建vuex多组件共享数据环境1.创建项目2.创建三个组件3.源代码三、vuex 的使用 - 创建仓库1.安装 vuex2.新建 `store/index.js` 专门存放 vuex3.创建仓库 `store/index.js`4 在 main.js 中导入挂载到 Vue 实例上5.…

密码学简介

密码学是研究信息安全的一门学科&#xff0c;主要涉及数据加密、解密和验证。其基本概念和术语包括&#xff1a; 1、明文与密文明文&#xff1a;未加密的原始数据。密文&#xff1a;经过加密处理的数据&#xff0c;通常是不可读的。 2、加密与解密加密&#xff1a;将明文转换…

Windows 部署非安装版Redis

1.下载Redis https://github.com/microsoftarchive/redis/releases 选择下载zip包&#xff0c;如Redis-x64-3.0.504.zip&#xff0c;并解压 2.启动非安装版redis服务 进入到redis目录&#xff0c;打开cmd 执行命令 redis-server.exe redis.windows.conf 3.登录redis客户端…

【Android】Gradle 7.0+ 渠道打包配置

声明 该配置主要解决打包apk/aab需要动态修改渠道字段,方便区分渠道上架国内商店。 暂不支持批量打包(7.4版本无法通过只修改outputFileName的形式批量处理) 因为构建时需要拷贝/创建Output,然后修改outputFileName才能处理批量打包,但拷贝/创建在高版本中失效了。 目前的…

一些单词的积累

情绪 2011年 2010 计算机模拟

4. STM32之TIM实验--输出比较(PWM输出,电机,四轴飞行器,智能车,机器人)--(实验2:PWM驱动舵机)

怎么区分伺服电机、舵机、步进电机&#xff1f; - 知乎 由图可知:通过调节占空比,来改变输出角度. 总结:其实舵机(齿轮控制减速机,直流电机,驱动器)就是一个山寨般的伺服电机(伺服电机可以说是一个系统:一个伺服电机包括控制器,编码器,电机,传感器等),其实很多单片机都配置了控…