Given a string s and a non-empty string p, find all the start indices of p's anagrams in s.
Strings consists of lowercase English letters only and the length of both strings s and p will not be larger than 20,100.
The order of output does not matter.
Example 1:
Input: s: "cbaebabacd" p: "abc" Output: [0, 6] Explanation: The substring with start index = 0 is "cba", which is an anagram of "abc". The substring with start index = 6 is "bac", which is an anagram of "abc".
Example 2:
Input: s: "abab" p: "ab" Output: [0, 1, 2] Explanation: The substring with start index = 0 is "ab", which is an anagram of "ab". The substring with start index = 1 is "ba", which is an anagram of "ab". The substring with start index = 2 is "ab", which is an anagram of "ab".
给一个字符串s和一个非空字符串p,找出s中所有的p的变位词的index。
解法:滑动窗口法,双指针。
Java:
class Solution {
public List<Integer> findAnagrams(String s, String p) {
int left = 0;
int right = 0;
int matchSize = p.length();
int[] map = new int[256];
List<Integer> res = new ArrayList<>();
// count the char number in p 计算p中各个字符的数量
for (char c:p.toCharArray()){
map[c] ++;
}
// build window 开始进行sliding window
while (right < s.length()){
// this char exists in p
// 如果当前的char是存在于p中,则目标大小matchsize就得减少
// 判断标准就是他的值不是为-1
if (map[s.charAt(right)] > 0)
matchSize --;
map[s.charAt(right)] --;
if (right - left == p.length()-1){
// check matchSize equals to 0
// 如果此时目标大小也是0,说明这就是需要的一个子串
if (matchSize == 0)
// add the left index
res.add(left);
// move left pointer to start new search
// 如果当这个字符原来是p中的话,现在移动指针需要还原以前原有的matchSize,开始新的搜索
if (map[s.charAt(left)] >= 0)
matchSize ++;
// 还原以前每个元素减去的1
map[s.charAt(left)]++;
left++;
}
right++;
}
return res;
}
}
Java:
public class Solution {
public List<Integer> findAnagrams(String s, String t) {
List<Integer> result = new LinkedList<>();
if(t.length()> s.length()) return result;
Map<Character, Integer> map = new HashMap<>();
for(char c : t.toCharArray()){
map.put(c, map.getOrDefault(c, 0) + 1);
}
int counter = map.size();
int begin = 0, end = 0;
int head = 0;
int len = Integer.MAX_VALUE;
while(end < s.length()){
char c = s.charAt(end);
if( map.containsKey(c) ){
map.put(c, map.get(c)-1);
if(map.get(c) == 0) counter--;
}
end++;
while(counter == 0){
char tempc = s.charAt(begin);
if(map.containsKey(tempc)){
map.put(tempc, map.get(tempc) + 1);
if(map.get(tempc) > 0){
counter++;
}
}
if(end-begin == t.length()){
result.add(begin);
}
begin++;
}
}
return result;
}
}
Python:
class Solution(object):
def findAnagrams(self, s, p):
"""
:type s: str
:type p: str
:rtype: List[int]
"""
result = []
cnts = [0] * 26
for c in p:
cnts[ord(c) - ord('a')] += 1
left, right = 0, 0
while right < len(s):
cnts[ord(s[right]) - ord('a')] -= 1
while left <= right and cnts[ord(s[right]) - ord('a')] < 0:
cnts[ord(s[left]) - ord('a')] += 1
left += 1
if right - left + 1 == len(p):
result.append(left)
right += 1
return result
Python: wo
class Solution(object):
def findAnagrams(self, s, p):
"""
:type s: str
:type p: str
:rtype: List[int]
"""
n = len(p)
counts = [0] * 26
for i in p:
counts[ord(i) - ord('a')] += 1
res = []
for i in xrange(len(s)):
counts[ord(s[i]) - ord('a')] -= 1
if i >= n:
counts[ord(s[i - n]) - ord('a')] += 1
if self.checkAllZero(counts):
res.append(i - n + 1)
return res
def checkAllZero(self, counts):
for i in counts:
if i != 0:
return False
return True
C++:
// Time: O(n)
// Space: O(1)
class Solution {
public:
vector<int> findAnagrams(string s, string p) {
vector<int> result;
if (p.empty() || s.empty()) {
return result;
}
vector<int> cnts(26);
for (const auto& c : p) {
++cnts[c - 'a'];
}
for (int left = 0, right = 0; right < s.length(); ++right) {
--cnts[s[right] - 'a'];
while (left <= right && cnts[s[right] - 'a'] < 0) {
++cnts[s[left++] - 'a'];
}
if (right - left + 1 == p.length()) {
result.emplace_back(left);
}
}
return result;
}
};
C++:
class Solution {
public:
vector<int> findAnagrams(string s, string p) {
if (s.empty()) return {};
vector<int> res, m(256, 0);
int left = 0, right = 0, cnt = p.size(), n = s.size();
for (char c : p) ++m[c];
while (right < n) {
if (m[s[right++]]-- >= 1) --cnt;
if (cnt == 0) res.push_back(left);
if (right - left == p.size() && m[s[left++]]++ >= 0) ++cnt;
}
return res;
}
};
类似题目:
[LeetCode] 242. Valid Anagram 验证变位词
[LeetCode] 567. Permutation in String 字符串中的全排列