
“巡逻的摄像机”。这台摄像机拍摄的,总是一段连续的画面(即一个“子数组”或“子串”)。我们的任务,就是通过移动这台摄像机,找到满足特定要求的那段最精彩、或是最短的画面。它的核心工作方式非常巧妙:通常有两个指针,一左一右,共同框定了摄像机的取景范围。右指针像是一个“开拓者”,负责向前探索,不断将新的元素纳入取景范围;而左指针则像是一个“优化师”,在条件满足时,负责从后面收紧范围,剔除不必要的元素,以寻找更优的解答。这一前一后、一扩一缩的配合,使得它不用重复检查所有可能的画面,就能高效地找到目标。
总而言之,滑动窗口的精髓在于,它通过两个指针的动态移动,聪明地维护着一个满足或试图满足条件的连续区间。它将很多看似复杂的问题时间复杂度从O(n²)降低到了高效的O(n),是处理连续子区间、子串问题的首选利器。


sum直到累加的和大于等于traget。(并初始化一个len为INT_MAX )在这个过程中
sum += nums[right++]就是在不断进窗口

left指针++,并且sum -= nums[right++],直到不满足sum大于等于traget这个结果。在这个过程中
sum -= nums[left++]就是在不断出窗口

sum>=traget的时候执行更新结果和出窗口的操作,最后返回最短的结果。
class Solution {
public:
int minSubArrayLen(int target, vector<int>& nums) {
int left = 0,right = 0;
int n = nums.size();
int len = INT_MAX;
int sum = 0;
for(right = 0;right < n;right++)
{
sum += nums[right];//进入窗口
while(sum >= target)//判断
{
len = min(len,right-left+1);//更新结果
sum -= nums[left++];//出窗口,减去左边left的值,因为右边固定,左边++,sum总会单调递减,直到sum<traget时,左边固定,右边++直到下一次循环
}
}
return len == INT_MAX ? 0 : len;
}
};//最短子数组长度 & 无重复字符最长子串
#include<iostream>
#include<vector>
using namespace std;
class Solution {
public:
int minSubArrayLen(int target, vector<int>& nums) {
int left = 0, right = 0;
int n = nums.size();
int len = INT_MAX;
int sum = 0;
for (right = 0; right < n; right++)
{
sum += nums[right];//进入窗口
while (sum >= target)//判断
{
len = min(len, right - left + 1);//更新结果
sum -= nums[left++];//出窗口,减去左边left的值,因为右边固定,左边++,sum总会单调递减,直到sum<traget时,左边固定,右边++直到下一次循环
}
}
return len == INT_MAX ? 0 : len;
}
};
int main()
{
vector<int> nums1 = { 2,3,1,2,4,3 };
int target1 = 7;
int result = Solution().minSubArrayLen(target1, nums1);
cout << "最短子数组长度: " << result << endl;
}


这样数组值记录对应字符在当前窗口内出现的次数

left标记当前无重复子串的起始位置(初始化为0),right用来探索新字符(初始化为0),扩展窗口右边界。
right逐个遍历字符串中字符,每遇到一个新字符,在哈希表中对应的位置计数加1(这里是进窗口) ,并更新它的最大长度(right-left+1)。
left指针右移(这就是出窗口),然后更新长度
hash[s[right]] > 1的时候出窗口的操作,然后再while循环外面执行更新结果和最后返回最短的结果。

class Solution {
public:
int lengthOfLongestSubstring(string s) {
int hash[128] = {0};
int left = 0,right = 0;
int n = s.size();
int ret = 0;
for(right = 0;right < n; right++)
{
hash[s[right]]++;//进窗口,ascall编码表,记录每个字符出现的次数
while(hash[s[right]] > 1)//判断
{
hash[s[left]]--;//出窗口,将模拟哈希中的字符次数--
left++;//更新下标,缩小窗口
}
ret = max(ret,right - left + 1);//更新结果
}
return ret;
}
};#include<iostream>
#include<string>
using namespace std;
class Solution {
public:
int lengthOfLongestSubstring(string s) {
int hash[128] = { 0 };
int left = 0, right = 0;
int n = s.size();
int ret = 0;
for (right = 0; right < n; right++)
{
hash[s[right]]++;//进窗口,ascall编码表,记录每个字符出现的次数
while (hash[s[right]] > 1)//判断
{
hash[s[left]]--;//出窗口,将模拟哈希中的字符次数--
left++;//更新下标,缩小窗口
}
ret = max(ret, right - left + 1);//更新结果
}
return ret;
}
};
int main()
{
string s1 = "abcabcbb";
int result = Solution().lengthOfLongestSubstring(s1);
cout << "无重复字符的最长子串长度: " << result << endl;
return 0;
}
在本篇文章中,我们通过「长度最小的子数组」与「无重复字符的最长子串」两个经典题目,深入理解了 滑动窗口 的核心思想与应用方式。
🚀 敬请期待下一篇: 【C++】优选算法必修篇之双指针实战:最大连续1的个数 & 将x减到0的最小操作数