力扣刷题(438. 找到字符串中所有字母异位词)滑动窗口

给定两个字符串 s 和 p,找到 s 中所有 p 的 异位词 的子串,返回这些子串的起始索引。不考虑答案输出的顺序。

异位词 指由相同字母重排列形成的字符串(包括相同的字符串)。

示例 1:

输入: s = "cbaebabacd", p = "abc"
输出: [0,6]
解释:
起始索引等于 0 的子串是 "cba", 它是 "abc" 的异位词。
起始索引等于 6 的子串是 "bac", 它是 "abc" 的异位词。


 示例 2:

输入: s = "abab", p = "ab"
输出: [0,1,2]
解释:
起始索引等于 0 的子串是 "ab", 它是 "ab" 的异位词。
起始索引等于 1 的子串是 "ba", 它是 "ab" 的异位词。
起始索引等于 2 的子串是 "ab", 它是 "ab" 的异位词。

思路:这是一道比较典型的滑动窗口的部分,首先用一个hashmap来纪录目标字符串中所有字符出现的次数,然后遍历字符串s。

如果没有在表中找到,说明该位置不对,那么直接把左指针移动到右指针位置,并将左指针走过的位置字符重新纪录到表里,左右指针再后移一位。

如果找到了,但是map中的值已经为0了,那就说明窗口中包含的该值过多,左指针右移一位,并更新map内容。此时右指针不动。

如果满足要求,就去判断当前窗口大小是否和目标字符串相同,相同的话,表面当前的窗口满足要求,纪录左指针,且左指针后移一位。右指针后移一位

class Solution {
public:
    vector<int> findAnagrams(string s, string p) {
        unordered_map<char,int> check;
        for(auto& k:p)++check[k];//初始化得到目标字符串中所有的字符内容
        //通过双指针/滑动窗口的方式来判断个数
        int ll=0,rr=0,num_s=s.size(),num_p=p.size();
        vector<int> ans;
        while(ll<=rr&&rr<num_s){
            if(check.find(s[rr])==check.end()){//该字符不存在与p中,直接让左指针移动到右指针
                while(ll<rr){//另表回复到初始状态
                    ++check[s[ll++]];
                }
                ++rr;
                ++ll;
            }
            else if(check[s[rr]]==0){//该字符存在p中,但是目前窗口中字符数量已满,左移一位
                ++check[s[ll++]];
            }
            else{//该位置的字符满足要求,在字符串内且目前的窗口中并没有过剩
                --check[s[rr]];
                if((rr-ll+1)==p.size()){//说明ll到rr这个区间内的字符串满足要求
                    ans.push_back(ll);//纪录满足要求的异位词的首部
                    //左指针左移,看下一个位置能否同样满足要求,可以的话就继续纪录,不可以的话就会
                    //在下一个位置让表回复到初始位置
                    ++check[s[ll++]];
                }
                ++rr;
            } 
        }
        return ans;
    }
};

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

yanzhe1

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值