给定两个字符串 s1 和 s2,写一个函数来判断 s2 是否包含 s1 的排列。
换句话说,第一个字符串的排列之一是第二个字符串的子串。
示例1:
输入: s1 = "ab" s2 = "eidbaooo"
输出: True
解释: s2 包含 s1 的排列之一 ("ba").
示例2:
输入: s1= "ab" s2 = "eidboaoo"
输出: False
注意:
输入的字符串只包含小写字母
两个字符串的长度都在 [1, 10,000] 之间
思路分析: 首先想到的是将第一个字符串全排列,然后用第二个逐个判断,但这样只能通过一部分就会超时。
正确的做法是使用滑动窗口的想法,使用两个哈希表的方式做统计(实际使用的是vector<int> a(128)),我们先来分别统计s1和s2中前n1个字符串中各个字符出现的次数,其中n1为字符串s1的长度,这样如果二者字符出现次数的情况完全相同,说明s1和s2中前n1的字符互为全排列关系,那么符合题意了,直接返回true。如果不是的话,那么我们遍历s2之后的字符,对于遍历到的字符,对应的次数加1,由于窗口的大小限定为了n1,所以每在窗口右侧加一个新字符的同时就要在窗口左侧去掉一个字符,每次都比较一下两个哈希表的情况,如果相等,说明存在
C ++代码如下:
class Solution {
public:
bool checkInclusion(string s1, string s2) {
int n1=s1.size();
int n2=s2.size();
if(n1 > n2) return false;
vector<int> map1(128),map2(128);
for(int i=0; i<n1; i++)
{
map1[s1[i]]++;
map2[s2[i]]++;
}
if(map1 == map2) return true;
for(int i=n1;i<s2.size();i++)
{
map2[s2[i]]++;
map2[s2[i-n1]]--;
if(map1 == map2) return true;
}
return false;
}
};