题目描述
给定两个字符串 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" 的异位词。
思路
双指针 + 滑动窗口 + 词频统计 + 哈希表
- 用哈希表来维护每个字符出现的次数。也用数组来进行词频统计,不要用排序法然后判断字符数组相等这样会超时。
- 第二个为滑动窗口版本。
代码
class Solution {
public List<Integer> findAnagrams(String s, String p) {
int n = s.length();
char pch[] = p.toCharArray();
int m = pch.length;
int[] c2 = new int[26];
for (int i = 0; i < m; i ++) {
c2[pch[i] - 'a'] ++;
}
char sch[] = s.toCharArray();
List<Integer> res = new ArrayList<>();
for (int i = 0; i + m - 1 < n; i ++ ) {
int t = 0, j = i;
int[] c1 = new int[26];
while (t < m) {
j ++;
t ++;
}
String sub = s.substring(i, j);
for (char c: sub.toCharArray()) {
c1[c - 'a'] ++;
}
if (check(c1, c2)) {
res.add(i);
}
}
return res;
}
public boolean check(int []c1, int []c2) {
for (int i = 0; i < 26; i ++ ) {
if (c1[i] != c2[i]) return false;
}
return true;
}
}
滑动窗口版本
class Solution {
public List<Integer> findAnagrams(String s, String p) {
List<Integer> res = new ArrayList<>();
int n = s.length();
int c1[] = new int[26];
int c2[] = new int[26];
int m = p.length();
for (int i = 0; i < m; i ++ ) {
c2[p.charAt(i) - 'a'] ++;
}
//遍历滑动窗口右端点
for (int i = 0, j = 0; i < n; i ++) {
c1[s.charAt(i) - 'a'] ++;
//如果超过滑动窗口
if (i - j + 1 > m) {
//删除超出的字符,同时移动左端点
c1[s.charAt(j) - 'a'] --;
j ++;
}
if (check(c1, c2)) {
res.add(j);
}
}
return res;
}
public boolean check(int c1[], int c2[]) {
for (int i = 0; i < 26; i ++ ) {
if (c1[i] != c2[i]) return false;
}
return true;
}
}