【滑动窗口:力扣438找到字符串中所有字母异位词】

题目链接:438. 找到字符串中所有字母异位词

这道题是让我们找s中所有p的异位词的字串,这里有一个新的概念"异位词",简单来说,异位词主要满足两个条件:

  • 两个单词由完全相同的字母组成(种类一致,个数一致)
  • 这些字母的排序方式不同

通过这两点我们就可以知道,互为异位词的两个字符串长度一定是相同的!因此我们可以用定长滑动窗口来做

定长滑动窗口

思路

在字符串s中维护一个滑动窗口,当窗口长度等于p.length()的时候,判断窗口内字母的种类跟个数是否等于p中字母的种类跟个数,如果满足条件,统计到答案中。

代码

java 复制代码
class Solution {
  public List<Integer> findAnagrams(String s, String p) {
    //统计 p 的每种字母的出现次数(题中说了字母只有小写字母)
    int [] cntP = new int[26];
    for(char c : p.toCharArray()) {
      cntP[c - 'a']++;
    }

    List<Integer> ans = new ArrayList<>();
    int [] cntS = new int[26];
    int n = p.length();
    int left = 0;
    for(int i = 0; i < s.length(); i++) {
      //统计s每种字母的出现次数
      cntS[s.charAt(i) - 'a']++;

      //窗口大小不够p的长度
      if(i - left + 1 < n) {
        continue;
      }

      //判断窗口内的每种字母的出现次数跟p的是否相同
      if(Arrays.equals(cntP, cntS)) {
        ans.add(left);
      }

      //移动左端点
      cntS[s.charAt(left) - 'a']--;
      left++;

    }
    return ans;
  }
}

除了这种方法,我们还可以使用不定长滑动窗口

不定长滑动窗口

思路

我们只需要枚举s子串的右端点,如果发现子串中某个字母的个数大于p中这种字母出现的次数,则缩小窗口(left右移),如果发现子串的长度等于p的长度,说明满足条件了,此时子串中每种字母的出现次数,等于 p 的每种字母的出现次数,统计到答案里面

这种方法理解的难点如下:

当子串长度跟p长度相等的时候,为什么能直接判断满足条件了,不会出现子串的其中一种字母的出现次数比 p 的小的情况吗?

不会。假设子串长度跟p的长度相等了,子串的某个字母的个数比p中这个字母的个数少,例如s是"aabbbc",p是"aabbcc"

此时子串中c的个数小于p中c的个数,但是由于长度相同,那么子串中一定有一个字母的个数大于p中这个字母的个数,这是矛盾的!

因此在"如果发现子串中某个字母的个数大于p中这种字母出现的次数,则缩小窗口"这个前提下,只要子串长度跟p的长度相等,子串中所有字母的种类和个数也都跟p中的字母种类个数相同!

代码

java 复制代码
class Solution {
  public List<Integer> findAnagrams(String s, String p) {
    //统计 p 的每种字母的出现次数(题中说了字母只有小写字母)
    int [] cntP = new int[26];
    for(char c : p.toCharArray()) {
      cntP[c - 'a']++;
    }

    List<Integer> ans = new ArrayList<>();
    int [] cntS = new int[26];
    int n = p.length();
    int left = 0;
    for(int i = 0; i < s.length(); i++) {
      //统计s每种字母的出现次数
      int c = s.charAt(i) - 'a';
      cntS[c]++;

      while(cntS[c] > cntP[c]) {
        cntS[s.charAt(left) - 'a']--;
        left++;
      }

      if(i - left + 1 == p.length()) {
        ans.add(left);
      }
    }
    return ans;
  }
}

如果这篇文章对你有帮助,欢迎点赞、评论、关注、收藏。你们的支持是我前进的动力!

相关推荐
kisshyshy10 小时前
🍦 雪糕、食堂、火车厢:三幅漫画吃透栈、队列与链表
javascript·算法
猿人谷17 小时前
不只是 CPU 阈值:STAR 如何用 GAT + Transformer 做容器级自动扩缩容?
人工智能·算法
复杂网络19 小时前
Stable Diffusion 视觉大模型微调技术深度调研
算法
复杂网络19 小时前
基于 Stable Diffusion 架构的视觉大模型代表性工作与原理深度解析
算法
MrZhao40019 小时前
Agent Loop 如何用 Hook 扩展:权限、日志与工具拦截
算法
MrZhao40019 小时前
Agent 为什么需要 Skills:别把所有知识都塞进 system prompt
算法
JieE2122 天前
LeetCode 101. 对称二叉树|JS 递归 + 迭代双解法,彻底搞懂镜像判断
javascript·算法
JieE2123 天前
LeetCode 56. 合并区间|超清晰 JS 图解思路,面试高频区间题
javascript·算法·面试
Jack203 天前
HarmonyOS开发中错误处理策略:网络异常统一处理
算法
小小杨树4 天前
读懂色彩:拍照调色不再难
算法·计算机视觉·配色