力扣 滑动窗口题目总结

Leetcode3.无重复字符的最长子串

思路:
这道题主要用到思路是:滑动窗口

什么是滑动窗口?

其实就是一个队列,比如例题中的 abcabcbb,进入这个队列(窗口)为 abc 满足题目要求,当再进入 a,队列变成了 abca,这时候不满足要求。所以,我们要移动这个队列!

如何移动?

我们只要把队列的左边的元素移出就行了,直到满足题目要求!

一直维持这样的队列,找出队列出现最长的长度时候,求出解!

时间复杂度:O(n)

力扣官方题解说明:

方法2:滑动窗口法

思路:

我们使用两个指针表示字符串中的某个子串(或窗口)的左右边界,其中左指针代表着上文中「枚举子串的起始位置」,而右指针即为上文中的 rk。

在每一步的操作中,我们会将左指针向右移动一格,表示 我们开始枚举下一个字符作为起始位置,然后我们可以不断地向右移动右指针,但需要保证这两个指针对应的子串中没有重复的字符。在移动结束后,这个子串就对应着 以左指针开始的,不包含重复字符的最长子串。我们记录下这个子串的长度;

在枚举结束后,我们找到的最长的子串的长度即为答案。

--判断重复字符:

在上面的流程中,我们还需要使用一种数据结构来判断 是否有重复的字符,常用的数据结构为哈希集合(即 C++ 中的 std::unordered_set,Java 中的 HashSet,Python 中的 set, JavaScript 中的 Set)。在左指针向右移动的时候,我们从哈希集合中移除一个字符,在右指针向右移动的时候,我们往哈希集合中添加一个字符。

至此,我们就完美解决了本题。

代码如下:

class Solution {//这道题主要用到思路是:滑动窗口// 定义一个方法来计算字符串中无重复字符的最长子串长度public int lengthOfLongestSubstring(String s) {//定义一个哈希集合,记录每个字符是否出现过(HashSet无序不可重复)Set<Character> occ = new HashSet<>(); //Charccter是Char的包装类int n = s.length();//右指针,初始值为-1,相当于我们在字符串的左边界的左侧,还没有开始移动int r = -1,res = 0;for(int i = 0; i < n; i ++){if(i != 0){//左指针向右移动一格,移除一个字符occ.remove(s.charAt(i-1));}//不断移动右指针,直到遇到重复字符或者到达字符串末尾while(r + 1 < n && !occ.contains(s.charAt(r+1))){//没到字符串末尾 && 不包含r+1-->将字符添加到集合中occ.add(s.charAt(r+1));//右指针右移++r;}//更新答案,取当前最大无重复子串的长度res = Math.max(res, r - i + 1);}// 返回结果,即最长无重复子串的长度return res;}
}

这段代码使用了滑动窗口技术,通过移动两个指针(左指针 i 和右指针 r)来寻找最长的无重复字符子串。具体操作如下:

  1. 初始化一个字符集合 occ 用于记录当前窗口中包含的字符。
  2. 定义右指针 的初始值为 -1,表示它还没有开始移动,并定义结果变量res 用于存储最长子串的长度。
  3. 通过外层 for 循环移动左指针 i,逐个处理字符串中的字符。
  4. 如果左指针 i 不是在初始位置,则移除左指针前一个位置的字符 s.charAt(i - 1),以确保当前窗口只包含无重复字符。
  5. 使用 while 循环不断右移右指针 r,扩展当前窗口,直到遇到重复字符或到达字符串末尾。每次右移时,将当前字符添加到集合 occ 中。
  6. 在每次扩展窗口后,更新结果 res 为当前窗口长度(r - i + 1)与之前最大长度中的较大值。
  7. 最终返回结果 res,即最长无重复字符子串的长度。

 补充知识点:

在Java中,Character 是一个包装类(wrapper class),它封装了一个基本类型 char 的值。Character 类提供了一些方法来操作字符数据类型。char 不同,Character 是一个类,可以用于泛型集合(如 SetList 等),因为这些集合只能包含对象,而不能包含基本数据类型。

下面是一些关于 Character 类的关键点:

  1. 封装基本类型 char

    • char 是一个基本数据类型,表示单个16位Unicode字符。
    • Character 类将 char 封装为对象,使其能够在需要对象的上下文中使用。
  2. 创建 Character 对象

    可以通过构造器或自动装箱(autoboxing)将 char 值转换为 Character 对象。
char ch = 'a';
Character characterObject = new Character(ch); // 使用构造器
Character characterObject2 = ch; // 自动装箱

     3.常用方法

  • Character 类提供了许多实用方法来处理字符。例如:
  • Character.isDigit(char ch): 检查字符是否为数字。
  • Character.isLetter(char ch): 检查字符是否为字母。
  • Character.toUpperCase(char ch): 将字符转换为大写。
  • Character.toLowerCase(char ch): 将字符转换为小写。

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

 

class Solution {// 定义一个方法来查找字符串 s 中所有是字符串 p 的字母异位词的子串的起始索引public List<Integer> findAnagrams(String s, String p) {// 获取字符串 s 和 p 的长度int n = s.length(), m = p.length();// 结果列表,用于存储所有符合条件的起始索引List<Integer> res = new ArrayList<>();// 如果 s 的长度小于 p 的长度,直接返回空列表if(n < m) return res;// 定义两个数组,用于记录 p 和 s 当前窗口中字符的频率int[] pCnt = new int[26];int[] sCnt = new int[26];// 初始化频率数组,将 p 和 s 前 m 个字符的频率记录到各自的数组中for(int i = 0; i < m; i++){pCnt[p.charAt(i) - 'a']++;sCnt[s.charAt(i) - 'a']++;}// 检查初始窗口(s 的前 m 个字符)是否与 p 匹配,如果匹配,将索引 0 加入结果列表if(Arrays.equals(sCnt, pCnt)){res.add(0);}// 遍历字符串 s,从索引 m 开始,每次向右滑动一个字符for(int i = m; i < n; i++){// 移除左边界字符的频率(滑动窗口左移)sCnt[s.charAt(i - m) - 'a']--;// 添加右边界字符的频率(滑动窗口右移)sCnt[s.charAt(i) - 'a']++;// 检查当前窗口是否与 p 匹配,如果匹配,将起始索引加入结果列表if(Arrays.equals(sCnt, pCnt)){res.add(i - m + 1);}}// 返回结果列表return res;}
}
  1. 初始化和边界检查

    • 获取 sp 的长度。
    • 如果 s 的长度小于 p 的长度,直接返回空列表,因为不可能有异位词。
  2. 频率数组初始化

    • 使用两个大小为 26 的数组 pCntsCnt 分别记录 ps 当前窗口中字符的频率。
    • 初始化频率数组,记录 psm 个字符(p 的长度)的频率。
  3. 初始窗口匹配检查

    • 检查 s 的前 m 个字符是否与 p 匹配,如果匹配,将索引 0 加入结果列表。
  4. 滑动窗口遍历

    • 从索引 m 开始遍历 s,每次滑动窗口右移一个字符:
      • 减少左边界字符的频率(移出窗口)。
      • 增加右边界字符的频率(加入窗口)。
      • 检查当前窗口是否与 p 匹配,如果匹配,将起始索引加入结果列表。
  5. 返回结果

    • 返回包含所有符合条件的起始索引的结果列表。

代码深度解释:

im 开始遍历到 n-1 时,代表滑动窗口的右边界每次右移一个字符,这两行代码分别是在这个过程中执行的操作。

  1. sCnt[s.charAt(i - m) - 'a']--;:这行代码是为了移除左边界字符的频率。在滑动窗口右移时,窗口的左边界向右移动一个字符,因此需要将移出窗口的左边界字符在 sCnt 数组中的计数减去 1。

  2. sCnt[s.charAt(i) - 'a']++;:这行代码是为了添加右边界字符的频率。在滑动窗口右移时,窗口的右边界向右移动一个字符,因此需要将新加入窗口的右边界字符在 sCnt 数组中的计数加上 1。

这两步操作保证了每次窗口的频率数组 sCnt 都能正确地反映当前窗口中字符的频率情况,以便后续比较是否与 p 的频率数组 pCnt 相同,从而判断是否满足异位词的条件。

class Solution {// 定义一个方法来查找字符串 s 中所有是字符串 p 的字母异位词的子串的起始索引public List<Integer> findAnagrams(String s, String p) {// 获取字符串 s 和 p 的长度int n = s.length(), m = p.length();// 结果列表,用于存储所有符合条件的起始索引List<Integer> res = new ArrayList<>();// 如果 s 的长度小于 p 的长度,直接返回空列表if (n < m) return res;// 定义两个数组,用于记录 p 和 s 当前窗口中字符的频率int[] pCnt = new int[26];int[] sCnt = new int[26];// 初始化 p 的频率数组,将 p 的每个字符的频率记录到 pCnt 数组中for (int i = 0; i < m; i++) {pCnt[p.charAt(i) - 'a']++;}// 定义左指针,初始值为 0int left = 0;// 遍历字符串 s,每次右指针右移一个字符for (int right = 0; right < n; right++) {// 获取右指针当前字符在字母表中的索引int curRight = s.charAt(right) - 'a';// 增加当前右指针字符在 sCnt 数组中的频率sCnt[curRight]++;// 如果当前右指针字符的频率超过了 p 中该字符的频率,需要缩小窗口while (sCnt[curRight] > pCnt[curRight]) {// 获取左指针当前字符在字母表中的索引int curLeft = s.charAt(left) - 'a';// 减少当前左指针字符在 sCnt 数组中的频率sCnt[curLeft]--;// 左指针右移一格left++;}// 如果当前窗口的长度等于 p 的长度,说明找到了一个异位词if (right - left + 1 == m) {// 将当前窗口的起始索引加入结果列表res.add(left);}}// 返回结果列表return res;}
}

这段代码使用了滑动窗口和字符频率计数技术来查找字符串 s 中所有是字符串 p 的字母异位词的子串的起始索引。具体操作如下:

  1. 初始化和边界检查

    • 获取 sp 的长度。
    • 如果 s 的长度小于 p 的长度,直接返回空列表,因为不可能有异位词。
  2. 频率数组初始化

    • 使用一个大小为 26 的数组 pCnt 记录 p 中每个字符的频率。
    • 使用一个大小为 26 的数组 sCnt 来记录当前滑动窗口中字符的频率。
  3. 滑动窗口遍历

    • 使用两个指针 leftright 表示当前滑动窗口的左右边界。
    • 右指针 right 从 0 开始遍历 s,每次右移一个字符,将该字符在 sCnt 数组中的频率加 1。
    • 如果当前字符的频率超过了 p 中该字符的频率,进入 while 循环,通过移动左指针 left 来缩小窗口,直到当前字符的频率不超过 p 中该字符的频率为止。
    • 如果当前窗口的长度等于 p 的长度,说明找到了一个异位词,将窗口的起始索引 left 加入结果列表。
  4. 返回结果

    • 返回包含所有符合条件的起始索引的结果列表。

 

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

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

相关文章

牛客NC334 字典序第K小【困难 10叉树 Java/Go/PHP/C++】,力扣 440. 字典序的第K小数字

题目 题目链接&#xff1a; https://www.nowcoder.com/practice/670c2bda374241d7ae06ade60de33e8b https://leetcode.cn/problems/k-th-smallest-in-lexicographical-order/description/ 本答案核心 10叉树, 数学规律Java代码 import java.util.*;public class Solution {…

大模型的灵魂解读:Anthropic AI的Claude3 Sonnet可解释性研究

大模型技术论文不断&#xff0c;每个月总会新增上千篇。本专栏精选论文重点解读&#xff0c;主题还是围绕着行业实践和工程量产。若在某个环节出现卡点&#xff0c;可以回到大模型必备腔调重新阅读。而最新科技&#xff08;Mamba,xLSTM,KAN&#xff09;则提供了大模型领域最新技…

Vue集成Iframe

一、应用场景&#xff0c;为什么要集成Iframe&#xff1f; 1、庞大项目拆分后&#xff0c;便于管理和部署&#xff0c;用集成Iframe的方法合并 2、避免功能重复开发&#xff0c;共用模块可单独开发为一个项目&#xff0c;既可独立部署&#xff0c;也可集成到中台系统 二、集成…

[算法][前缀和] [leetcode]724. 寻找数组的中心下标

题目地址 https://leetcode.cn/problems/find-pivot-index/description/ 题目描述 代码 class Solution {public int pivotIndex(int[] nums) {int total Arrays.stream(nums).sum();//前缀和int prefixSum 0;int len nums.length;for(int i 0;i<len;i){if (i-1>0){p…

小猪APP分发:一站式托管服务,轻松玩转应用市场

在当今移动应用爆炸式增长的时代&#xff0c;开发者们面临的挑战不再仅限于创意的火花和代码的实现&#xff0c;更在于如何让精心打造的应用快速触达广大用户。这正是小猪APP分发www.appzhu.net应运而生的背景——作为一个全面、高效的APP托管服务分发平台&#xff0c;它为开发…

基于PHP的物业管理的设计与实现

第1章 绪论... 1 1.1 研究背景与意义... 1 1.2 国内外发展现状... 2 第2章 关键技术介绍... 3 2.1 PHP语言... 3 2.2 MySQL数据库... 3 2.3 Zend框架... 4 2.4 B/S架构... 4 第3章 系统需求分析... 5 3.1 可行性分析... 5 3.1.1 技术可行性分析... 5 3.1.2 经济可行…

金职优学:分析央国企面试如何通关?

在当今竞争激烈的就业市场中&#xff0c;中央和国有企业&#xff08;以下简称“央国企”&#xff09;的面试机会对求职者来说是非常有吸引力的。这些企业通常拥有稳定的发展前景、良好的薪酬福利和广阔的职业发展空间。但是&#xff0c;要想成功通过央国企的面试&#xff0c;求…

探索Python编程世界:从基础到实战

新书上架~&#x1f447;全国包邮奥~ python实用小工具开发教程http://pythontoolsteach.com/3 欢迎关注我&#x1f446;&#xff0c;收藏下次不迷路┗|&#xff40;O′|┛ 嗷~~ 目录 一、Python语言简介与动态特性 代码示例&#xff1a;动态类型与变量命名 二、Python应用领…

vue 表格表头展示不下,显示。。。;鼠标悬浮展示全部

vue 表格表头展示不下&#xff0c;显示。。。&#xff1b;鼠标悬浮展示全部 <templateslot-scope"scope"slot"header"><span:title"临时证券类型"style"white-space:nowrap">{{ 临时证券类型 }}</span></templa…

Terminal Web终端基础(Web IDE 技术探索 二)

Terminal是web终端技术&#xff0c;类似cmd命令窗口&#xff0c;Webcontainer 中推荐使用的是Xterm.js&#xff0c;这里就不细说Xterm.js 的使用了&#xff0c;我们使用第三方库来实现&#xff08;原生确实有点难用&#xff09;。 vue-web-terminal 一个由 Vue 构建的支持多内容…

【设计模式】JAVA Design Patterns——Bytecode(字节码模式)

&#x1f50d;目的 允许编码行为作为虚拟机的指令 &#x1f50d;解释 真实世界例子 一个团队正在开发一款新的巫师对战游戏。巫师的行为需要经过精心的调整和上百次的游玩测试。每次当游戏设计师想改变巫师行为时都让程序员去修改代码这是不妥的&#xff0c;所以巫师行为以数据…

AcW木棒-XMUOJ恢复破碎的符咒木牌-DFS与剪枝

题目 思路 话不多说&#xff0c;直接上代码 代码 /* AcW木棒-XMUOJ恢复破碎的符咒木牌 搜索顺序&#xff1a;从小到大枚举最终的长度 len从前往后依次拼每根长度为len的木棍 优化&#xff1a; 1.优化搜索顺序&#xff1a;优先选择深度短的来搜索&#xff0c;故从大到小去枚…

【系统分析师】WEB开发-案例

文章目录 1、WEB开发涉及内容1.1 负载均衡技术1.2 数据库读写分离1.3 缓存 缓解读库压力1.4 CDN1.5 WEB应用服务器1.6 整体结构1.6 相关技术1.6.1 redis相关(集群、持久化等)1.6.2 XML与JSON1.6.3 REST1.6.4 响应式web设计1.6.5 关于中台1.6.6 Web系统分层 1、WEB开发涉及内容 …

Python--面向对象

面向对象⭐⭐ 1. 面向对象和面向过程思想 面向对象和面向过程都是一种编程思想,就是解决问题的思路 面向过程&#xff1a;POP(Procedure Oriented Programming)面向过程语言代表是c语言面向对象&#xff1a;OOP(Object Oriented Programming)常见的面向对象语言包括:java c g…

19c数据库19.9以下dg切换打开hang住问题

原主库发起切换请求&#xff0c;原主库正常切换数据库角色&#xff0c;但原从库无法正常打开数据库&#xff0c;尝试关闭重启&#xff0c;依旧无法解决问题。 查看切换过程中原从库数据库后台日志&#xff0c;发现数据库一直不断重试清理 SRLs&#xff0c; 后台alert日志&…

力扣HOT100 - 21. 合并两个有序链表

解题思路&#xff1a; class Solution {public ListNode mergeTwoLists(ListNode list1, ListNode list2) {ListNode dum new ListNode(0), cur dum;while (list1 ! null && list2 ! null) {if (list1.val < list2.val) {cur.next list1;list1 list1.next;} els…

基本IO接口

引入 基本输入接口 示例1 示例2&#xff1a;有数据保持能力的外设 #RD端由in指令控制&#xff1a;将数据由端口传输到CPU内存中 #CS244信号由译码电路实现 示例3&#xff1a; a)图中由于输出端口6有连接到端口1&#xff0c;当开关与端点1闭合时期间&#xff0c;仍能维持3端口…

插件:NGUI

一、版本 安装完毕后重启一下即可&#xff0c;否则可能创建的UI元素不生效 二、使用 Label文字 1、创建Canvs 2、只有根节点的这些脚本全部展开才能鼠标右键创建UI元素 3、选择字体 Sprite图片 1、选择图集 2、选择图集中的精灵 Panel容器 用来装UI的容器&#xff0c;一般UI…

《计算机网络微课堂》2-5 信道的极限容量

本节课我们介绍信道极限容量的有关问题。 我们都知道信号在传输过程中会受到各种因素的影响&#xff0c;如图所示&#xff0c;这是一个数字信号&#xff0c;‍‍当它通过实际的信道后&#xff0c;波形会产生失真&#xff0c;当失真不严重时&#xff0c;在输出端‍‍还可根据以失…

Redis实现热点数据排行榜或游戏积分排行榜

数据库中的某张表中存储着文章的浏览量&#xff0c;或者点赞数等&#xff0c;或者游戏积分等数据...... 这些数据的更新在redis中完成&#xff0c;并定时同步到mysql数据库中。 而如果要对这些数据进行排序的话&#xff1a; Redis中的Sorted Set(有序集合)非常适合用于实现排…