跳到主要内容
极客日志极客日志
首页博客AI提示词GitHub精选代理工具
搜索
|注册
博客列表
C++算法

滑动窗口算法详解与 LeetCode 高频题目解题模板

综述由AI生成滑动窗口是一种解决数组或字符串子区间问题的高效算法,能将时间复杂度从 O(n²) 优化至 O(n)。通过 LeetCode 长度最小的子数组和最长无重复子串两个经典案例,对比暴力解法与滑动窗口策略,阐述连续性、单调移动、局部更新及最优性保证四大核心理念。提供了完整的 C++ 代码实现与执行逻辑分析,帮助读者掌握双指针技巧,提升算法面试解题能力。

草莓泡芙发布于 2026/3/21更新于 2026/5/1119 浏览

滑动窗口算法详解

**滑动窗口(Sliding Window)**是解决数组或字符串子区间问题的利器。无论是'最小覆盖子串',还是'长度最小的子数组',滑动窗口都能以 O(n) 的时间复杂度优雅解决。

1. 为什么要学滑动窗口?

LeetCode 209. 长度最小的子数组

给定一个含有 n 个正整数的数组和一个正整数 target。找出该数组中满足其和 ≥ target 的长度最小的连续子数组,并返回其长度。

输入: target = 7, nums = [2,3,1,2,4,3]
输出: 2
解释: 子数组 [4,3] 是该条件下的长度最小的子数组。

暴力解法思路
  1. 枚举所有可能的子数组
  2. 计算每个子数组的和
  3. 找出满足条件的最短子数组
// 暴力解法 O(n²)
int minSubArrayLen_brute(int target, vector<int>& nums) {
    int n = nums.size();
    int minLen = INT_MAX;
    // 枚举所有起始位置
    for (int i = 0; i < n; i++) {
        int sum = 0;
        // 枚举所有结束位置
        for (int j = i; j < n; j++) {
            sum += nums[j];
            if (sum >= target) {
                minLen = min(minLen, j - i + 1);
                break; // 找到了就 break,因为后面的更长
            }
        }
    }
    return minLen == INT_MAX ? 0 : minLen;
}
暴力解法的问题分析

当 n=10000 时,需要约 1 亿次操作,严重超时。

滑动窗口的优化思路

观察:窗口的连续性

假设我们有一个窗口 [2,3,1] 和为 6 < 7。

  • 传统做法: 丢弃整个窗口,从 3 重新开始。
  • 聪明做法: 为什么不只丢弃左边的 2,保留 [3,1] 呢?

当前窗口:[2,3,1] 和=6 < 7 扔掉左边:移除 2 → [3,1] 和=4 加入右边:加入 2 → [3,1,2] 和=6 加入右边:加入 4 → [3,1,2,4] 和=10 ≥ 7 这样避免了重复计算 [3,1] 的和!

滑动窗口的核心思想

就像相机取景框,只移动边界,不重新拍摄整个画面。

int minSubArrayLen_sliding(int target, vector<int>& nums) {
    int left = 0;       // 窗口左边界
    int sum = 0;        // 当前窗口的和
    int minLen = INT_MAX; // 最小长度
    int n = nums.size();
    
    for (int right = 0; right < n; right++) {
        // 1. 加入右边元素(扩大窗口)
        sum += nums[right];
        
        // 2. 当窗口满足条件时,尝试收缩窗口
        while (sum >= target) {
            // 更新最小长度
            minLen = min(minLen, right - left + 1);
            // 收缩窗口:移除左边元素
            sum -= nums[left];
            left++;
        }
        // 3. 如果窗口不满足条件,继续扩大窗口(下一轮循环)
    }
    return minLen == INT_MAX ? 0 : minLen;
}

先向右扩大,当 sum 足够时,再从左缩小。

2. 滑动窗口的核心思想

四大核心理念

1. 连续性原则

只处理连续的子序列,不是任意的子集。

  • ✅ [1,2,3,4] 中的 [2,3,4](连续,可以用滑动窗口)
  • ❌ [1,2,3,4] 中的 [1,3,4](不连续,不能用滑动窗口)
2. 单调移动原则

左右指针只能单向移动(通常向右)。

int left = 0, right = 0;
while (right < n) {
    // right 只会增加
    // left 只会增加或不变
    // 永远不会回头!
}
3. 局部更新原则

窗口从 [2,3,1] 滑动到 [3,1,2]。

  • 传统方法:重新计算 3+1+2 = 6
  • 滑动窗口:利用已有的 2+3+1=6,然后 new_sum = old_sum - 离开的元素 + 新进入的元素 = 6 - 2 + 2 = 6
  • 只计算变化的部分!
4. 最优性保证原则

通过合理的收缩规则,确保不会错过最优解。

while (窗口不满足条件) {
    收缩窗口; // 剔除"坏"元素
}
// 此时窗口是当前右边界下的"局部最优"

3. LeetCode 题目实战

1. 代码实现:最长无重复子串

class Solution {
public:
    int lengthOfLongestSubstring(string s) {
        unordered_set<char> charSet;
        int left = 0, maxLen = 0;
        for (int right = 0; right < s.size(); right++) {
            while (charSet.find(s[right]) != charSet.end()) {
                charSet.erase(s[left]);
                left++;
            }
            charSet.insert(s[right]);
            maxLen = max(maxLen, right - left + 1);
        }
        return maxLen;
    }
};

set 是 C++ STL 中的一种关联容器,它存储唯一的元素(不允许重复),并且元素会自动排序。

  • #include <set>:有序集合(基于红黑树实现)
  • #include <unordered_set>:无序集合(基于哈希表实现)
大致思路
  1. 用一个 unordered_set 来记录当前窗口里的字符,保证里面没有重复。
  2. 用双指针 left、right 表示窗口的左右边界,left 初始为 0。
  3. 用 maxLen 记录遍历过程中最长的无重复子串长度。
  4. right 从左到右遍历字符串,每次尝试把 s[right] 加入窗口。
  5. 如果发现 s[right] 已经在 set 里(重复了),就不断从左边 erase 字符、left++,直到把重复的那个字符彻底移出窗口。
  6. 把当前 s[right] 加入 set,更新窗口长度,维护 maxLen。
  7. 遍历结束后,maxLen 就是答案。
重点讲解 while 循环

它的作用是:当遇到重复字符时,持续移动左指针,直到移除那个重复字符。

例子:s = "abcbda"

我们用滑动窗口 [left, right] 来跟踪当前无重复字符的子串。

步骤分析:

  1. 初始状态:left = 0, charSet = {}
  2. right = 0:'a' 不在集合中,直接加入 → charSet = {'a'}, 长度=1
  3. right = 1:'b' 不在集合中,直接加入 → charSet = {'a', 'b'}, 长度=2
  4. right = 2:'c' 不在集合中,直接加入 → charSet = {'a', 'b', 'c'}, 长度=3
  5. right = 3:'b' 已经在集合中!
    • 进入 while 循环
    • 第一次循环:移除 s[left](即 s[0] = 'a'),left = 1
    • 检查 'b' 还在集合中吗?是的,因为集合现在是 {'b', 'c'}
    • 第二次循环:移除 s[left](即 s[1] = 'b'),left = 2
    • 现在集合是 {'c'},'b' 不在集合中了
    • 退出 while 循环
    • 加入 'b',集合变为 {'c', 'b'},长度 = right - left + 1 = 3 - 2 + 1 = 2
  6. right = 4:'d' 不在集合中,直接加入 → charSet = {'c', 'b', 'd'}, 长度=3

后续步骤依此类推,最终返回最大长度。

目录

  1. 滑动窗口算法详解
  2. 1. 为什么要学滑动窗口?
  3. LeetCode 209. 长度最小的子数组
  4. 暴力解法思路
  5. 暴力解法的问题分析
  6. 滑动窗口的优化思路
  7. 观察:窗口的连续性
  8. 滑动窗口的核心思想
  9. 2. 滑动窗口的核心思想
  10. 四大核心理念
  11. 1. 连续性原则
  12. 2. 单调移动原则
  13. 3. 局部更新原则
  14. 4. 最优性保证原则
  15. 3. LeetCode 题目实战
  16. 1. 代码实现:最长无重复子串
  17. 大致思路
  18. 重点讲解 while 循环
  19. 例子:s = "abcbda"
  • 💰 8折买阿里云服务器限时8折了解详情
  • GPT-5.5 超高智商模型1元抵1刀ChatGPT中转购买
  • 代充Chatgpt Plus/pro 帐号了解详情
  • 🤖 一键搭建Deepseek满血版了解详情
  • 一键打造专属AI 智能体了解详情
极客日志微信公众号二维码

微信扫一扫,关注极客日志

微信公众号「极客日志V2」,在微信中扫描左侧二维码关注。展示文案:极客日志V2 zeeklog

更多推荐文章

查看全部
  • GitHub Desktop 中文汉化方法:界面本地化配置指南
  • 基于无人机的多模态目标检测高多样性基准与基线
  • 医疗AI新范式:数理模型重构传统大模型面临的挑战
  • DeepSeek 深度使用指南:提示词工程与本地知识库搭建
  • PHP 低代码流程建模的 5 个致命误区与优化实践
  • 仓库管理系统前端开发:主区域查询与重置功能
  • 网络安全入门教程:从零开始掌握基础技术与工具
  • 大模型应用开发指南:LangChain 核心概念与实战解析
  • 基于开源模型的成人内容过滤合规解决方案
  • Windows 安装原生 Codex CLI 配置 AI 代码助手
  • OpenClaw 本地部署指南:从零搭建 AI 助理
  • 柔性抓取“慧眼”:MEMS 3D 视觉如何识别无序堆叠钣金件?
  • VMware 虚拟机 CentOS 磁盘扩容指南(解决 growpart 报错与 LVM 扩容)
  • 基于Xilinx FPGA的RISC-V五级流水线CPU设计与实现
  • PyTorch 文本引导图像生成与 Stable Diffusion 实践
  • Flutter 与 Kotlin 对比:移动应用开发选型指南
  • 文心一言5.0 Preview模型能力观察:LMArena文本任务实测
  • 零基础网络安全入门指南:学习路线与实战建议
  • Python 集成 IPIDEA 网页抓取 API 实现 eBay 数据采集实战
  • uniapp 判断运行设备类型(安卓、苹果、鸿蒙、微信小程序、H5)

相关免费在线工具

  • 加密/解密文本

    使用加密算法(如AES、TripleDES、Rabbit或RC4)加密和解密文本明文。 在线工具,加密/解密文本在线工具,online

  • Gemini 图片去水印

    基于开源反向 Alpha 混合算法去除 Gemini/Nano Banana 图片水印,支持批量处理与下载。 在线工具,Gemini 图片去水印在线工具,online

  • Base64 字符串编码/解码

    将字符串编码和解码为其 Base64 格式表示形式即可。 在线工具,Base64 字符串编码/解码在线工具,online

  • Base64 文件转换器

    将字符串、文件或图像转换为其 Base64 表示形式。 在线工具,Base64 文件转换器在线工具,online

  • Markdown转HTML

    将 Markdown(GFM)转为 HTML 片段,浏览器内 marked 解析;与 HTML转Markdown 互为补充。 在线工具,Markdown转HTML在线工具,online

  • HTML转Markdown

    将 HTML 片段转为 GitHub Flavored Markdown,支持标题、列表、链接、代码块与表格等;浏览器内处理,可链接预填。 在线工具,HTML转Markdown在线工具,online