滑动窗口算法精解:用C++高效解决子串/子数组问题
创作时间:
作者:
@小白创作中心
滑动窗口算法精解:用C++高效解决子串/子数组问题
引用
CSDN
1.
https://m.blog.csdn.net/muzibuku/article/details/146169125
滑动窗口算法是一种高效处理连续子序列问题的算法,通过维护一个动态变化的窗口区间,将暴力算法的O(n²)时间复杂度优化到O(n)。本文将从滑动窗口算法的基本概念出发,通过多个经典应用场景的代码实现,深入解析其核心思想和适用场景。
一、滑动窗口的本质:智能伸缩的取景框
滑动窗口算法如同一个智能摄像机,在数据序列中寻找最完美的"镜头"。它通过维护一个动态变化的窗口区间,将暴力算法的O(n²)时间复杂度优化到O(n),特别适用于解决连续子序列相关问题。我们将从三大经典应用场景深入解析:
二、滑动窗口的三大核心要素
2.1 窗口维护三要素
int slidingWindowTemplate(vector<int>& nums, int k) {
int left = 0, right = 0; // 双指针定义窗口边界
unordered_map<int, int> window; // 窗口状态记录
int valid = 0; // 满足条件的指标
while (right < nums.size()) {
// 1. 扩展右边界
int c = nums[right++];
window[c]++;
if (window[c] == k) valid++;
// 2. 收缩左边界条件
while (valid == target) {
// 3. 更新最优解
res = min(res, right - left);
int d = nums[left++];
if (window[d] == k) valid--;
window[d]--;
}
}
return res;
}
2.2 算法复杂度分析
问题类型 | 暴力复杂度 | 滑动窗口复杂度 | 优化倍数 |
|---|---|---|---|
最长无重复子串 | O(n²) | O(n) | n倍 |
最小覆盖子串 | O(n²) | O(n+m) | n/m倍 |
长度最小子数组 | O(n²) | O(n) | n倍 |
三、基础应用:最长无重复子串(LeetCode 3)
3.1 哈希表+双指针实现
int lengthOfLongestSubstring(string s) {
vector<int> map(128, -1); // ASCII直接寻址
int maxLen = 0, left = 0;
for (int right = 0; right < s.size(); ++right) {
if (map[s[right]] >= left)
left = map[s[right]] + 1; // 跳跃收缩
map[s[right]] = right; // 记录最新位置
maxLen = max(maxLen, right - left + 1);
}
return maxLen;
}
// 输入:"abcabcbb" → 输出:3("abc")
算法亮点:
- ASCII直接映射:O(1)时间查询
- 左边界跳跃:避免逐次收缩
- 实时更新最大值:无需额外存储
四、进阶应用:最小覆盖子串(LeetCode 76)
4.1 多条件窗口维护
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, start = 0, len = INT_MAX;
while (right < s.size()) {
char c = s[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;
}
char d = s[left++];
if (need.count(d)) {
if (window[d] == need[d]) valid--;
window[d]--;
}
}
}
return len == INT_MAX ? "" : s.substr(start, len);
}
// 输入:s = "ADOBECODEBANC", t = "ABC" → 输出:"BANC"
关键技巧:
- 需求字典(need):记录目标字符频次
- 有效计数(valid):跟踪匹配字符数
- 动态记录最优解:实时更新最小窗口
五、滑动窗口的四大变种
5.1 固定窗口大小
vector<double> findAverages(vector<int>& arr, int k) {
vector<double> res;
int sum = 0;
for (int right = 0; right < arr.size(); ++right) {
sum += arr[right];
if (right >= k-1) {
res.push_back(sum / (double)k);
sum -= arr[right - k + 1];
}
}
return res;
}
// 输入:[1,3,2,6,-1], k=3 → 输出:[2.0, 3.67, 2.33]
5.2 最多K次替换后的最长子串
int characterReplacement(string s, int k) {
vector<int> count(26);
int maxCount = 0, maxLen = 0;
int left = 0;
for (int right = 0; right < s.size(); ++right) {
maxCount = max(maxCount, ++count[s[right]-'A']);
while (right-left+1 - maxCount > k) {
count[s[left++]-'A']--;
}
maxLen = max(maxLen, right-left+1);
}
return maxLen;
}
// 输入:s="AABABBA", k=1 → 输出:4("AABA"→"AAAA")
六、性能优化与陷阱规避
6.1 常见性能陷阱
陷阱类型 | 错误示例 | 优化方案 |
|---|---|---|
无效窗口收缩 | 每次移动左边界1位 | 跳跃收缩 |
冗余状态计算 | 每次重新统计窗口内容 | 增量更新 |
哈希表查询瓶颈 | 频繁使用count()检查 | 数组直接寻址 |
指针更新顺序错误 | 先移动指针再更新状态 | 先处理当前状态再移动 |
6.2 高级优化技巧
// 优化点1:数组替代哈希表
vector<int> count(128, 0); // 处理ASCII字符
// 优化点2:跳跃收缩左边界
left = max(left, lastPos + 1);
// 优化点3:提前终止循环
if (maxLen == s.size()) break;
七、滑动窗口的六大应用场景
- 子串搜索:最小覆盖子串、字母异位词
- 数组统计:最大平均值、乘积小于K
- 流式处理:数据流中位数、最近K元素
- 字符串分析:重复DNA序列、回文子串
- 时间序列:股票最佳时机、日程安排
- 资源分配:课程安排、任务调度
结语:滑动窗口的智慧之光
滑动窗口算法展现了算法设计的三个核心哲学:
- 时空权衡:用空间记录状态换取时间效率
- 增量思维:避免重复计算已有信息
- 最优剪枝:及时排除无效解可能
当你在LeetCode遇到以下题目时,滑动窗口将是你的破题利器:
- 长度最小的子数组
- 替换后的最长重复字符
- 字符串的排列
掌握滑动窗口的精髓,你将能优雅地解决大量子序列问题,如同拥有解开数据迷宫的万能钥匙。记住:优秀的算法不是魔法,而是对问题模式的深刻洞察与巧妙建模。
热门推荐
鹿城墨池坊:温州必打卡文艺圣地!
《蛋仔派对》搞笑视频合集:笑到肚子疼!
《蛋仔派对》PK《糖豆人》,谁是休闲游戏之王?
预防流感,哪些措施最有效?
季节性流感如何传播?有何规律?复旦团队在Science发文
从心理学角度看:固执的背后竟是原生家庭?
吴翘胡子:固执不通的代表
秋冬必备!苹果橙子水温暖上线
煮苹果水的正确打开方式:功效、配方与饮用建议
冬至时节,煮一碗暖心的苹果水
自律博主天花板:北京海淀小学生
短期旅行回来后,有哪些整理照片和回忆的创意方式?
驾考体检检查哪些项目
贵州非遗 | 酒香之处,便是阿穴人家
木耳是否含胶原蛋白?吃木耳对心血管有好处?木耳5大功效报你知!
世界与中国 | 文化产业是中国经济发展的重要组成部分
韩国农产品价格猛涨20%:复杂零售体系推高物价
盘点奥特曼经典主题曲:哪首最燃?
如何建立房东和租客之间的良好关系
千年绸都日日新——吴江盛泽探索人文与经济共生共荣之道
秋季种荠菜,你get了吗?
小虫子,大危险!夏季防蚊虫叮咬,这份攻略不可少!
苹果水登顶热搜!教你轻松融入生活
苹果煮水:健康新宠!
怀疑自己登革热?有这些症状要马上就医
春季和秋季:荠菜种植的最佳时机!
春天来了!教你如何挖到最嫩的荠菜
经济危机下,我们都在同一条船上
《蛟龙行动》点映票房破1700万!观众热议现代海军战术应用
《蛟龙行动》幕后花絮引爆网络热议:从1:1潜艇到演员们的极限挑战