滑动窗口算法框架:
void slidingWindow(string s,string t)
{
unordered_map<char,int> need,window;
//need 题目中要求的字符串
//window窗口滑动过程中拥有的字符串
for(char c : t) need[c]++;
int left = 0,right=0;
int valid=0;
//其中valid记录的是有几个相等的字符和目标字符串(t)
while(right<s.size()) //终止条件
{
//c是即将要入窗口的字符
char c=s[right];
right++;//右移动窗口
...........//自己对窗口中信息进行更新,需要添加的代码
while(window needs shrink) //判断左侧窗口是否需要收缩
{
char d=s[left];
left++;//收缩 左移动窗口
.........//自己对窗口中信息进行更新,需要添加的代码
}
}
例如(LeetCode最小覆盖子串):
string minWindow(string s, string t) {
unordered_map<char, int> need, window;
for (char c : t) need[c]++;
int left = 0, right = 0;
int valid = 0;
// 记录最小覆盖子串的起始索引及长度
int start = 0, len = INT_MAX;
while (right < s.size()) {
// c 是将移入窗口的字符
char c = s[right];
// 右移窗口
right++;
// 进行窗口内数据的一系列更新
if (need.count(c)) {
window[c]++;
if (window[c] == need[c])
valid++;
}
// 判断左侧窗口是否要收缩
while (valid == need.size()) {
// 在这里更新最小覆盖子串,在这里更新最小串,使用的是两个下标的方式,目的是为了方便截取
if (right - left < len) {
start = left;
len = right - left;
}
// d 是将移出窗口的字符
char d = s[left];
// 左移窗口
left++;
// 进行窗口内数据的一系列更新
if (need.count(d)) {
if (window[d] == need[d])
valid--;
window[d]--;
}
}
}
// 返回最小覆盖子串
return len == INT_MAX ?
"" : s.substr(start, len);
}