题目
给你两个字符串 s1 和 s2 ,写一个函数来判断 s2 是否包含 s1 ****的排列。如果是,返回 true ;否则,返回 false 。
换句话说,s1 的排列之一是 s2 的 子串 。
示例 1:
输入: s1 = "ab" s2 = "eidbaooo"
输出: true
解释: s2 包含 s1 的排列之一 ("ba").
滑动窗口
我们需要维护一个窗口,不断的滑动,过程中对窗口进行缩小或者扩大。 大体框架代码为:
class Solution {
public boolean checkInclusion(String s1, String s2) {
Map<Character,Integer> need = new HashMap<>();
Map<Character,Integer> window = new HashMap<>();
for(char c : s1.toCharArray()){
need.put(c,need.getOrDefault(c,0)+1);
}
int left = 0,right = 0,count = 0;
while(right < s2.length()){
char c = s2.charAt(right);
right++;
//进行窗口的更新
//判断是否需要收缩
while(left < right && window needs shrink){
char d = s2.charAt(left);
left++;
}
}
}
return false;
}
}
对于此题,我们需要注意的是,s1为进行判断的字符串,因此need里放入的为s1, 当right划过包含字串的字母时应该对窗口进行更新
if(need.containsKey(c)){
window.put(c,window.getOrDefault(c,0)+1);
if(need.get(c).equals(window.get(c))){
count++;
}
}
同时由于是字串问题,所以答案只能出自当左右指针差值与字串相等时,因为判断收缩条件即为
right - left == s1.length()
此时收缩内的函数应该进行更新,对left进行移动,保持左右差值为字符的长度,因此整体代码为
class Solution {
public boolean checkInclusion(String s1, String s2) {
Map<Character,Integer> need = new HashMap<>();
Map<Character,Integer> window = new HashMap<>();
for(char c : s1.toCharArray()){
need.put(c,need.getOrDefault(c,0)+1);
}
int left = 0,right = 0,count = 0;
while(right < s2.length()){
char c = s2.charAt(right);
right++;
if(need.containsKey(c)){
window.put(c,window.getOrDefault(c,0)+1);
if(need.get(c).equals(window.get(c))){
count++;
}
}
//只有在right - left == need.size()的情况下才有可能有符合条件的解
while(right - left == s1.length()){
if(count == need.size())
return true;
char d = s2.charAt(left);
left++;
if(need.containsKey(d)){
if(need.get(d).equals(window.get(d))){
count--;
}
window.put(d,window.getOrDefault(d,0)-1);
}
}
}
return false;
}
}