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

思路:滑动窗口。

对于主串s和子串p,设n是子串p的长度。

一、方法一:定长滑动窗口。枚举主串s的所有长为n的子串s',在滑动的同时维护s'中的每种字母的出现次数。如果s'的每种字母的出现次数和p的每种字母的出现次数都相同,那么s'是p的异位词,并把s'的左端点下标加入答案。

附代码:

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']++; //统计p的字母
        }
        List<Integer> res = new ArrayList<>();
        int[] cnts = new int[26]; //统计s的长度为p.length()的子串s'的每种字母的出现次数
        for(int right = 0;right < s.length();right++){
            cnts[s.charAt(right) - 'a']++; //右端点字母进入窗口
            int left = right - p.length() + 1;
            if(left < 0){ //窗口长度不足p.length()
                continue;
            }
            if(Arrays.equals(cnts,cntp)){ //s'和p的每种字母的出现次数都相同
                res.add(left); //s'的左端点下标加入答案
            }
            cnts[s.charAt(left) - 'a']--; //左端点字母离开窗口
        }
        return res;
    }
}

二、方法二:不定长滑动窗口。枚举子串s'的右端点,如果发现s'其中一种字母的出现次数大于p的这种字母的出现次数,则右移s'的左端点(缩小窗口)。如果发现s'的长度等于p的长度,则说明s'的每种字母的出现次数等于p的每种字母的出现次数,即s'是p的异位词。

证明:内层循环结束后,s'的每种字母的出现次数都小于等于p的每种字母的出现次数。如果s'的其中一种字母的出现次数比p的小,那么s'的长度必然小于p的长度。所以只要s'的长度等于p的长度,就说明s'的每种字母的出现次数和p的每种字母的出现次数都相同,就说明s'是p的异位词,于是把s'的左端点下标加入答案。

代码实现时,可以把cnts和cntp合并成一个cnt:

(1)对于p的字母c,把cnt[c]加一。

(2)对于s的字母c,把cnt[c]减一。

(3)如果cnt[c]小于0,则说明窗口中的字母c的个数比p的多,右移左端点。

附代码:

java 复制代码
class Solution {
    public List<Integer> findAnagrams(String s, String p) {
        //统计p的每种字母的出现次数
        int[] cnt = new int[26];
        for(char c : p.toCharArray()){
            cnt[c - 'a']++;
        }
        List<Integer> res = new ArrayList<>();
        int left = 0;
        for(int right = 0;right < s.length();right++){
            int c = s.charAt(right) - 'a';
            cnt[c]--; //右端点字母进入窗口
            while(cnt[c] < 0){ //字母c太多了
                cnt[s.charAt(left) - 'a']++; //左端点字母离开窗口
                left++;
            }
            if(right - left + 1 == p.length()){ //s'和p的每种字母的出现次数都相同
                res.add(left); //s'的左端点下标加入答案
            }
        }
        return res;
    }
}
相关推荐
sheji34169 小时前
【开题答辩全过程】以 基于协同过滤算法电影个性化推荐系统设计与实现为例,包含答辩的问题和答案
算法
uesowys9 小时前
Apache Spark算法开发指导-Random forest classifier
算法·随机森林·spark
仙俊红11 小时前
LeetCode487周赛T2,删除子数组后的最终元素
数据结构·算法
-dzk-17 小时前
【代码随想录】LC 59.螺旋矩阵 II
c++·线性代数·算法·矩阵·模拟
风筝在晴天搁浅17 小时前
hot100 78.子集
java·算法
Jasmine_llq17 小时前
《P4587 [FJOI2016] 神秘数》
算法·倍增思想·稀疏表(st 表)·前缀和数组(解决静态区间和查询·st表核心实现高效预处理和查询·预处理优化(提前计算所需信息·快速io提升大规模数据读写效率
超级大只老咪18 小时前
快速进制转换
笔记·算法
m0_7066532318 小时前
C++编译期数组操作
开发语言·c++·算法
故事和你9118 小时前
sdut-Java面向对象-06 继承和多态、抽象类和接口(函数题:10-18题)
java·开发语言·算法·面向对象·基础语法·继承和多态·抽象类和接口
qq_4232339018 小时前
C++与Python混合编程实战
开发语言·c++·算法