【滑动窗口:力扣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;
  }
}

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

相关推荐
武子康12 分钟前
调查研究-151 Slack vs Jira:区别、使用指南与团队选择方法
人工智能·科技·深度学习·ai·职场和发展·jira·slack
8Qi832 分钟前
LeetCode 23. 合并 K 个升序链表 —— 小顶堆(PriorityQueue)
数据结构·算法·leetcode·链表·
QiLinkOS1 小时前
《打破“用爱发电”:一种基于 Gitee 与时间戳的开源权益分配机制探索》
c语言·数据结构·c++·科技·算法·gitee·开源
松间听晚1 小时前
Agentic RL 环境和代码学习:以HGPO为例
算法
智者知已应修善业2 小时前
【51单片机用T0定时器方式1,实现0.5S的时间间隔实现第一次一个灯亮、第二次二个灯亮,直到全部灯亮,然后重复整个过程】2023-12-29
c++·经验分享·笔记·算法·51单片机
小许同学记录成长2 小时前
几何体编辑与布尔运算
算法·无人机
fanged2 小时前
简单看看3A算法2(TODO)
算法
智者知已应修善业2 小时前
【51单片机4位静态数码管显示1234】2023-11-14
c++·经验分享·笔记·算法·51单片机
♡すぎ♡2 小时前
镜面 IBL 预过滤贴图的计算
算法·计算机图形学·贴图·pbr
Lsk_Smion3 小时前
力扣实训 _ [200].岛屿数量
算法·leetcode·深度优先