LeetCode——滑动窗口(进阶)

LeetCode——滑动窗口(进阶)

文章目录

相关例题

904. 水果成篮

题目描述

你正在探访一家农场,农场从左到右种植了一排果树。这些树用一个整数数组 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 = [1,2,3,2,2] 输出:4 解释:可以采摘 [2,3,2,2] 这四棵树。 如果从第一棵树开始采摘,则只能采摘 [1,2] 这两棵树。 

示例 4:

输入:fruits = [3,3,3,1,2,1,1,2,3,3,4] 输出:5 解释:可以采摘 [1,2,1,1,2] 这五棵树。 

提示:

  • 1 <= fruits.length <= 105
  • 0 <= fruits[i] < fruits.length

题目分析

这个题目其实还是一个使用滑动窗口的题目,我们分析题意可以直到我们这里要维护的窗口是一个最长的数字类型不大于2的窗口,并且题目也说明了这里不会存在不连续的情况(一旦你走到某棵树前,但水果不符合篮子的水果类型,那么就必须停止采摘。),所以这题可以使用滑动窗口。

实现思路

一开始我们需要定义两个指针都是指向开头的,并且我们这里为了获取数字类型的个数并且出窗口方便,这里需要定义一个哈希表(这里不可以用集合,想想为什么),然后就是窗口的操作了。

我们首先需要将右指针的值入窗口,如果类型数(也就是我们哈希表的大小)大于2了,我们就开始出窗口,也就是将我左指针指向的值的哈希值--,同时减到了0还要删除出哈希表(这里就是为什么不能用集合了),出了条件之后就是我们的返回值更新。

最后就是将我们的值返回,其实这里只要数组大小大于了1就有返回值,但是我们这里还是按照了之前的写法(bushi )。

实现代码

classSolution{public:inttotalFruit(vector<int>& fruits){ unordered_map<int,int> h;int len =-1;int size = fruits.size();for(int i =0, j =0; j < size; j++){ h[fruits[j]]++;while(h.size()>2){ h[fruits[i]]--;if(h[fruits[i]]==0){ h.erase(fruits[i]);} i++;} len =max(len, j - i +1);}return len ==-1?0: len;}};

438. 找到字符串中所有字母异位词

题目描述

给定两个字符串 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 仅包含小写字母

题目分析

这个题目比较容易陷进找有效开头这个点里面去,其实这个题目不需要我们找有效开头,只需要维护一个满足条件的区间就可以了,所以这个题目还是用的滑动窗口来维持着这样一个区间(一开始没想到正确的解法)。

实现思路

我们这里其实就是维持子串的字符哈希值(也就是这个字符出现的个数)要大于等于我们的窗口里面对应字符的哈希值,同时我们的窗口大小也不能大于我们的子串的大小。

首先我们需要预处理我们的子串对应字符的哈希值,同时设置我们需要的一些变量。

然后就是定义两个指针指向开头来遍历我们的父串,如果当前的字符出现次数要小于等于我们的子串对应的字符,我们就将我们的计数++,然后就是我们的越界判断,如果我们的长度大于了子串的长度,我们就判断当前父串中左指针的字符出现次数是不是要小于子串对应字符出现的次数,如果是小于等于就将我们的计数--,这里同时也考虑了我们的非法的情况(非法的字符不参与计数计数--,但是要和参与的数字一起排除),然后就是判断我们的计数是不是满足题意了并加入到我们的返回数组中,最后就是返回我们的数组了。

实现代码

classSolution{public: vector<int>findAnagrams(string s, string p){ vector<int> ret;int hash1[26]={0};int hash2[26]={0};for(auto e:p) hash2[e-'a']++;int n=s.size(),m=p.size();for(int left=0,right=0,count=0;right<n;right++){if(++hash1[s[right]-'a']<=hash2[s[right]-'a']) count++;if(right-left+1>m){if(hash1[s[left]-'a']--<=hash2[s[left]-'a']) count--; left++;}if(count==m) ret.push_back(left);}return ret;}};

串联所有单词的子串

题目描述

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

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 由小写英文字母组成

题目分析

这是一个困难题,但是如果我们有了上面那个题目作为支撑,这个题目的思路还是比较好想的,我们这里其实可以吧每一个单词看成是一个一个的字母,这个问题就变成了求字符串里面所有的字母的异位词,处理对象从字符变成了单词而已,但是这个题目还是有一个比较重要的优化思路需要了解。

实现思路

我们这里首先就是确定我们的开始位置,我们这里不可以直接从0开始,这样会出现有的情况没考虑到,同时这里直接设置每个字符都作为一次我们的开头也是不可以的(会有很多重复的结果),这里我们应该值设置前words中单词的长度个字符,这样就可以不重不漏,我们这里还是画图理解:

在这里插入图片描述

搞定了开头的字符之后,我们接下来的实现就是和之前实现【找到字符串中所有字母异位词】的逻辑一样了,只不过这里有一些细节需要注意,详细细节可以看代码。

代码实现

classSolution{public: vector<int>findSubstring(string s, vector<string>& words){ vector<int> ret;int n = s.size(), len = words[0].size(), m = words.size(); unordered_map<string,int> hash1;for(auto e : words) hash1[e]++;for(int i =0; i < len; i++){ unordered_map<string,int> hash2;for(int left = i, right = i, count =0; right + len <= n;right += len)// 这里的实现要是等于n,否则这个样例就会出错s = "wordgoodgoodgoodbestword" words = ["word","good","best","good"]{ string in = s.substr(right, len);if(++hash2[in]<= hash1[in]) count++;if(right - left +1> len * m){ string out = s.substr(left,len);if(hash2[out]--<= hash1[out]) count--; left+=len;}if(count == m) ret.push_back(left);}}return ret;}};

最小覆盖子串

题目描述

给你一个字符串 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 由英文字母组成

**进阶:**你能设计一个在 o(m+n) 时间内解决此问题的算法吗?

题目分析

这个题目是一个困难题,这里使用的还是滑动窗口的知识,我们维护的窗口是一个最小长度并且覆盖了所有目标字符的子串,我们要做的就是维护好这个子串并且实时更新出最小的答案。

实现思路

我们这里首先需要预处理我们需要覆盖串中字符的出现次数,处理好之后我们还需要定义一个长度的最大值,然后就是我们窗口的维护了,这个时候我们需要定义两个指针来指向我们的开头位置,首先将我们的右指针指向的字符出现的次数++后判断是不是属于我们的目标串,属于就将计数++;当我们的计数等于我们目标串的大小的时候,我们就可以来更新我们的最小长度了和我们的返回串的开头了,与此同时我们需要从左边出窗口并判断是不是影响我们的窗口的条件;最后就是我们的返回值返回了,如果还是最大值说明没有更新,于是返回空串,反之将我们返回串返回。

实现代码

classSolution{public: string minWindow(string s, string t){int n = s.size();int k = t.size(); unordered_map<char,int> h1; unordered_map<char,int> h2;for(auto& e : t){ h1[e]++;}int cnt =0;int start =0;int len =0x3f3f3f3f;for(int l =0, r =0; r < n; r++){if(++h2[s[r]]<= h1[s[r]]) cnt++;while(cnt == k){if(len > r - l +1){ len = r - l +1; start = l;}if(h2[s[l]]--== h1[s[l++]]) cnt--;}}if(len ==0x3f3f3f3f)return"";else{ string ret(s.begin()+ start, s.begin()+ start + len);return ret;}}};

Read more

【前端】Vue 组件开发中的枚举值验证:从一个Type属性错误说起

【前端】Vue 组件开发中的枚举值验证:从一个Type属性错误说起

🌹欢迎来到《小5讲堂》🌹 🌹这是《小程序》系列文章,每篇文章将以博主理解的角度展开讲解。🌹 🌹温馨提示:博主能力有限,理解水平有限,若有不对之处望指正!🌹 👨💻 作者简介 🏆 荣誉头衔:2024博客之星Top14 | ZEEKLOG博客专家 | 阿里云专家博主 🎤 经历:曾多次进行线下演讲,亦是 ZEEKLOG内容合伙人 以及 新星优秀导师 💡 信念:“帮助别人,成长自己!” 🚀 技术领域:深耕全栈,精通 .NET Core (C#)、Python、Java,熟悉主流数据库 🤝 欢迎交流:无论是基础概念还是进阶实战,都欢迎与我探讨! 目录 * 前言 * 解决过程 * 一、错误场景还原 * 1.1 错误发生的位置 * 1.2 常见的触发场景 * 二、深入理解 Vue

By Ne0inhk

Android WebRTC VAD 实战指南:从原理到避坑

快速体验 在开始今天关于 Android WebRTC VAD 实战指南:从原理到避坑 的探讨之前,我想先分享一个最近让我觉得很有意思的全栈技术挑战。 我们常说 AI 是未来,但作为开发者,如何将大模型(LLM)真正落地为一个低延迟、可交互的实时系统,而不仅仅是调个 API? 这里有一个非常硬核的动手实验:基于火山引擎豆包大模型,从零搭建一个实时语音通话应用。它不是简单的问答,而是需要你亲手打通 ASR(语音识别)→ LLM(大脑思考)→ TTS(语音合成)的完整 WebSocket 链路。对于想要掌握 AI 原生应用架构的同学来说,这是个绝佳的练手项目。 从0到1构建生产级别应用,脱离Demo,点击打开 从0打造个人豆包实时通话AI动手实验 Android WebRTC VAD 实战指南:从原理到避坑 在语音通话或语音识别应用中,如何让设备"聪明&

By Ne0inhk
【踩坑记录】使用 Layui 框架时解决 Unity WebGL 渲染在 Tab 切换时黑屏问题

【踩坑记录】使用 Layui 框架时解决 Unity WebGL 渲染在 Tab 切换时黑屏问题

【踩坑记录】使用 Layui 框架时解决 Unity WebGL 渲染在 Tab 切换时黑屏问题 在开发 Web 应用时,尤其是集成了 Unity WebGL 内容的页面,遇到一个问题:当 Unity WebGL 渲染内容嵌入到一个 Tab 中时,切换 Tab 后画面会变黑,直到用户点击黑屏区域,才会恢复显示。 这个问题通常是因为 Unity 渲染在 Tab 切换时被暂停或未能获得焦点所致。 在本文中,我们将介绍如何在使用 Layui 框架时,通过监听 Tab 切换事件并强制 Unity WebGL 渲染恢复,来解决这一问题。 1. 问题描述 当 Unity WebGL 内容嵌入到页面中的多个

By Ne0inhk
Flutter for OpenHarmony:web 拥抱 Web 标准的桥梁(Wasm GC 与 DOM 互操作) 深度解析与鸿蒙适配指南

Flutter for OpenHarmony:web 拥抱 Web 标准的桥梁(Wasm GC 与 DOM 互操作) 深度解析与鸿蒙适配指南

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net 前言 随着 Flutter 3.x 全面拥抱 Wasm(WebAssembly),Dart 团队推出了全新的 package:web 来取代老旧的 dart:html。 package:web 是基于最新的 JS Interop 机制构建的,它不仅性能更好,而且兼容 Wasm GC 标准。 虽然这个库通过名字看是为 “Web” 平台的,但对于 OpenHarmony 开发者来说,了解它有着特殊的意义: 1. 混合开发:鸿蒙原生支持 ArkWeb (WebView),在 Flutter 中通过 JS互操作与 Web 页面交互是常见需求。 2.

By Ne0inhk