LeetCode Hot100 - 滑动窗口篇

前言

挑战一个月刷完力扣的hot100,记录一下每题的思路~

这次是滑动窗口相关的题目

(1)3. 无重复字符的最长子串

双指针+变长 滑动窗口。若已知一个不重复子串(l到r),右指针扩充一个(r+1),只可能r+1 与某个k重复,因为l到k的字符不可能再产生新的更长不重复子串,于是将l循环收缩,直至新字符不重复(即l收缩到k+1),。然后r再继续往后扩充找新子串

java 复制代码
class Solution {
    Set<Character> set = new HashSet<>();
    public int lengthOfLongestSubstring(String s) {
        int res=0;
        // 每次右指针加一个
        for(int l=0,r=0; r<s.length(); r++){
            // 若当前元素已包含,循环缩短左边界,直至不在set中
            while(set.contains(s.charAt(r)))set.remove(s.charAt(l++));
            set.add(s.charAt(r)); // 添加当前元素
            res = Math.max(res,r-l+1); // 保留最长串
        }
        return res;
    }
}

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

定长 滑动窗口。窗口长度始终为pLen,使用长度26的数组记录滑动窗口和p的字符数量,不断后移滑动窗口比较是否为异位词

java 复制代码
class Solution {
    int[] sCount = new int[26], pCount = new int[26]; // 计数
    List<Integer> res = new ArrayList<>();
    public List<Integer> findAnagrams(String s, String p) {
        int sLen=s.length(), pLen=p.length();
        if(sLen<pLen)return res; // 长度不足
        for(int i=0; i<pLen; i++){ // 初始化计数数组
            ++sCount[s.charAt(i)-'a'];
            ++pCount[p.charAt(i)-'a'];
        }
        if(Arrays.equals(sCount,pCount))res.add(0); // 异位词
        for(int i=0;i<sLen-pLen;i++){ // 滑动窗口长度始终为pLen
            --sCount[s.charAt(i)-'a'];
            ++sCount[s.charAt(i+pLen)-'a'];
            if(Arrays.equals(sCount,pCount))res.add(i+1);
        }
        return res;
    }
}

②双指针+变长 滑动窗口。用count记录p与滑动窗口差值 ,计数p初始化count。用双指针遍历,r右移取一位,维护count不小于0,循环收缩左边界。若滑动窗口长度与p同,且count不小于0,即每个元素相同,记录l

java 复制代码
class Solution {
    int[] count = new int[26]; // 计数,p与滑动窗口差值
    List<Integer> res = new ArrayList<>();
    public List<Integer> findAnagrams(String s, String p) {
        int sLen=s.length(), pLen=p.length();
        if(sLen<pLen)return res; // 长度不足
        for(char c:p.toCharArray()) ++count[c-'a']; // 初始化计数数组
        for(int l=0,r=0; r<sLen; r++){ // 双指针遍历
            --count[s.charAt(r)-'a']; // 取r
            // 循环收缩左边界,维持每个元素不小于0
            while(count[s.charAt(r)-'a']<0) ++count[s.charAt(l++)-'a'];
            // 长度相同,且每个元素不小于0,即个数相同
            if(r-l+1==pLen)res.add(l);
        }
        return res;
    }
}

总结

①滑动窗口常用于连续子串查找

②无重复字符的最长子串。用set和双指针+变长 滑动窗,每次移动右指针,若有重复就循环收缩左边界,直至不重复,再往右继续判断

③找到字符串中所有字母异位词。第一种定长 滑动窗口,滑动窗口长度始终与p同,用数组统计滑动窗口和p字符个数 ,若个数相同就记录结果,不断移动窗口判断。第二种双指针+变长 滑动窗口,与上题类似,count小于0就循环收缩左边界,若窗口长度等于p就记录l

相关推荐
鸽鸽程序猿4 分钟前
【算法】【优选算法】宽搜(BFS)中队列的使用
算法·宽度优先·队列
Jackey_Song_Odd4 分钟前
C语言 单向链表反转问题
c语言·数据结构·算法·链表
Watermelo6178 分钟前
详解js柯里化原理及用法,探究柯里化在Redux Selector 的场景模拟、构建复杂的数据流管道、优化深度嵌套函数中的精妙应用
开发语言·前端·javascript·算法·数据挖掘·数据分析·ecmascript
乐之者v13 分钟前
leetCode43.字符串相乘
java·数据结构·算法
A懿轩A1 小时前
C/C++ 数据结构与算法【数组】 数组详细解析【日常学习,考研必备】带图+详细代码
c语言·数据结构·c++·学习·考研·算法·数组
古希腊掌管学习的神1 小时前
[搜广推]王树森推荐系统——矩阵补充&最近邻查找
python·算法·机器学习·矩阵
云边有个稻草人1 小时前
【优选算法】—复写零(双指针算法)
笔记·算法·双指针算法
半盏茶香1 小时前
在21世纪的我用C语言探寻世界本质 ——编译和链接(编译环境和运行环境)
c语言·开发语言·c++·算法
忘梓.2 小时前
解锁动态规划的奥秘:从零到精通的创新思维解析(3)
算法·动态规划
tinker在coding4 小时前
Coding Caprice - Linked-List 1
算法·leetcode