给定两个字符串 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;
}
};