力扣 滑动窗口题目总结

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']++;
        }

        // 定义左指针,初始值为 0
        int 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. 返回结果

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

 

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

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

相关文章

2024年上半年软件系统架构师论文【回忆版】

文章目录 考试时间考试地点案例分析1、微服务架构的优点和缺点2、质量属性的6个元素3、分布式锁 Redis的缺点4、MongoDB 存储矢量图的优势 论文回忆版论文一、论单元测试的设计与应用论文二、论大数据模型的设计与应用论文三、论模型驱动的架构设计及应用论文四、论云原生运维的…

LeetCode 279 —— 完全平方数

阅读目录 1. 题目2. 解题思路3. 代码实现 1. 题目 2. 解题思路 此图利用动态规划进行求解&#xff0c;首先&#xff0c;我们求出小于 n n n 的所有完全平方数&#xff0c;存放在数组 squareNums 中。 定义 dp[n] 为和为 n n n 的完全平方数的最小数量&#xff0c;那么有状态…

esp32-idf 开发踩坑记录

现象 直接使用原始命令编译idf.py build 但是提示idf 版本错误 卸载旧版本 编译出错build 问题 然后删除编译文件后&#xff0c;重新编译&#xff0c;还是出错 解决方法1 最后发现是因为项目所在文件夹有中文目录&#xff0c;把项目迁移到英文目录后&#xff0c;重新编译&a…

【Python 对接QQ的接口(二)】简单用接口查询【等级/昵称/头像/Q龄/当天在线时长/下一个等级升级需多少天】

文章日期&#xff1a;2024.05.25 使用工具&#xff1a;Python 类型&#xff1a;QQ接口 文章全程已做去敏处理&#xff01;&#xff01;&#xff01; 【需要做的可联系我】 AES解密处理&#xff08;直接解密即可&#xff09;&#xff08;crypto-js.js 标准算法&#xff09;&…

同名在线查询系统微信小程序源码下载支持多种流量主,附带系统教程

同名在线查询系统微信小程序源码下载支持多种流量主这是一款支持查询同名的一款微信小程序 该款小程序支持多种查询模式 重名查询&#xff0c;热度查询&#xff0c;概率香查询 源码免费下载地址抄笔记(chaobiji.cn)

arcgisPro将一个图层的要素复制到另一个图层

1、打开两个图层&#xff0c;如下&#xff0c;其中一个图层中有两个要素&#xff0c;需要将其中一个要素复制到另一个图层中&#xff0c;展示如下&#xff1a; 2、选中待复制要素&#xff0c;点击复制按钮&#xff0c;如下&#xff1a; 3、下拉粘贴按钮列表&#xff0c;选择【选…

普通函数的参数中的auto

2.1 普通函数的参数中的auto 从c14起&#xff0c;lambda可以使用auto占位符声明或者定义参数: auto printColl [] (const auto& coll) // generic lambda{ for (const auto& elem : coll) {std::cout << elem << \n;}} 只要支持Lambda 内部的操作&…

java图书电子商务网站的设计与实现源码(springboot+vue+mysql)

风定落花生&#xff0c;歌声逐流水&#xff0c;大家好我是风歌&#xff0c;混迹在java圈的辛苦码农。今天要和大家聊的是一款基于springboot的图书电子商务网站的设计与实现。项目源码以及部署相关请联系风歌&#xff0c;文末附上联系信息 。 项目简介&#xff1a; 图书电子商…

推荐一款自助分析的财务分析软件:奥威BI软件

奥威BI软件是一款支持多维度动态自助分析的软件&#xff0c;预设了智能财务分析方案&#xff0c;提供内存行列计算模型解决财务指标计算难题&#xff0c;界面简洁&#xff0c;以点击、拖曳操作为主&#xff0c;十分适合没有IT背景的财务人做财务分析。因此也经常有人说奥威BI软…

tcp协议介绍,协议段格式(端口号,首部长度,窗口大小,序号,确认序号,6个标志位),流量控制,确认应答机制,捎带应答,三次握手的双方认知不一致问题

目录 tcp协议 介绍 传输控制协议 图解 全双工 缓冲区 控制 tcp协议段格式 数据在不同层的名称 图解 ​编辑 端口号 首部长度 窗口大小 -- 引入 前提 流量控制 确认应答机制 窗口大小 -- 介绍 序号 -- 引入 确认应答机制的进一步探讨 如果应答丢失 捎带应…

基于Android studio 使用SQLite数据库完成登录注册功能——保姆级教程

&#x1f345;文章末尾有获取完整项目源码方式&#x1f345; 点击快捷传送地址&#xff1a; 保姆级教学——制作登陆注册功能页面 目录 一、准备工作 二、创建相关文件 三、页面布局 四、DabaHelper帮助类的编写 五、RegisterActivity注册页面 六、LoginActivity登录页面…

clion读取文件设置为读取当前目录下的文件

1.问题 使用vs读取文件时一切正常 但是同样的代码在clion中无法正常执行 原因 原因&#xff1a;clion的源文件找不到input.txt文件的位置 需要设置工作目录&#xff0c;例如此时input.txt在当前目录下&#xff0c;那么就设置 设置当前文件的工作目录为$FileDir$即可&am…

通过 NIO + 多线程 提升硬件设备与系统的数据传输性能

一、项目展示 下图&#xff08;模拟的数据可视化大屏&#xff09;中数据是动态显示的 二、项目简介 描述&#xff1a;使用Client模拟了硬件设备&#xff0c;比如可燃气体浓度检测器。Client通过Socket与Server建立连接&#xff0c;Server保存数据到txt文件&#xff0c;并使用W…

操作系统总结4----死锁的处理策略总结

目录 2.4.2 死锁的处理策略-----预防死锁 &#xff08;1&#xff09;知识总览 &#xff08;2&#xff09;破环互斥条件 &#xff08;3&#xff09;破环不剥夺条件 &#xff08;4&#xff09;破环求情和保持条件 &#xff08;5&#xff09;破环循环等待条件 总结 2.4.3 死…

广告圈策划大师课:活动策划到品牌企划的深度解析

对于刚接触营销策划的新人来说&#xff0c;在这个知识密集型行业里生存&#xff0c;要学习非常多各种意思相近的概念&#xff0c;常常让人感到头疼&#xff0c;难以区分。 这里对这些策划概念进行深入解析&#xff0c;帮助您轻松理清各自的含义和区别。 1. 活动策划&#xff…

操作抖音小店一直不出单怎么办?只需要做好这两点就可以了!

大家好&#xff0c;我是电商小V 最近很多新手小伙伴来咨询我说自己操作抖音小店&#xff0c;自己的店铺长时间不出单应该怎么办&#xff1f;今天咱们就来详细的说一下&#xff0c; 咱们要清楚的就是自己的店铺不出&#xff0c;只需要咱们做好这两点就可以了&#xff0c; 第一点…

华为机考入门python3--(29)牛客29-字符串加解密

分类&#xff1a;字符变换 知识点&#xff1a; 字符是字母 char.isalpha() 字符是小写字母 char.islower() 字符是数字 char.isdigit() b变C chr((ord(b) - ord(a) 1) % 26 ord(A)) 题目来自【牛客】 # 加密 def encrypt_string(s):result ""for ch…

C++的AVL树

目录 基本概念 插入的语言分析 LL右旋 RR左旋 额外结论及问题1 LR左右旋 RL右左旋 额外结论及问题2 插入结点 更新bf与判断旋转方式 旋转代码实现 准备工作一 LL右旋的实现 RR左旋的实现 准备工作二 LR左右旋的实现 RL右左旋的实现 完整代码 基本概念 1、…

抖音小店新规重磅来袭!事关店铺流量!商家的福音来了?

大家好&#xff0c;我是喷火龙。 就在前两天&#xff0c;抖店发布了新规&#xff0c;我给大家总结了一下&#xff0c;无非就是两点。 第一点&#xff1a;保证金下调&#xff0c;一证开多店。 第二点&#xff1a;新品上架破10单&#xff0c;有流量扶持。 咱来细细的解读&…

无人机助力光伏项目测绘建模

随着全球对可再生能源需求的不断增长&#xff0c;光伏项目作为其中的重要一环&#xff0c;其建设规模和速度都在不断提高。在这一背景下&#xff0c;如何高效、准确地完成光伏项目的测绘与建模工作&#xff0c;成为了行业发展的重要课题。近年来&#xff0c;无人机技术的快速发…