
 送给大家一句话:
那脑袋里的智慧,就像打火石里的火花一样,不去打它是不肯出来的。——莎士比亚
滑动窗口入门
- 认识滑动窗口
 - Leetcode 209. 长度最小的子数组
 - 题目描述
 - 算法思路
 
- Leetcode 3. 无重复字符的最长子串
 - 题目描述
 - 算法思路
 
- Leetcode 1004. 最大连续1的个数 III
 - 题目描述
 - 算法思路
 
- 总结
 - 送给大家一句话:
 - Thanks♪(・ω・)ノ谢谢阅读!!!
 - 下一篇文章见
 
今天我学习了滑动窗口的算法思路,接下来请与我一起看看吧!!!
认识滑动窗口
滑动窗口问题可以说是一种特殊的双指针问题,通常用于解决以下类型的问题:
- 连续子数组或子字符串问题:例如,找出一个数组中连续元素和最大或最小的子数组,或者在字符串中找到一个包含特定字符的最短子字符串。
 - 固定窗口大小问题:当窗口大小固定时,我们可以通过移动窗口来遍历整个数组或字符串,并记录所需的统计信息。
 - 可变窗口大小问题:在某些情况下,窗口的大小可能会根据特定条件而变化。这需要我们在遍历过程中动态地调整窗口的大小。
 
滑动窗口算法的基本思想是使用双指针(有时也可能使用更多指针)来表示窗口的边界。在每一步中,我们可以根据特定条件来移动窗口的边界,并更新所需的统计信息。
看这些定义是真无法想象出来哦怎么个滑动窗口的,下面我们一起来做题吧:
Leetcode 209. 长度最小的子数组
题目描述

 看这个题目还是很好理解的,只需要我们找到和大于target的连续子数组,我们来看第一个样例target = 7, nums = [2,3,1,2,4,3] 显然4,3是最小的子数组。接下来分析一下算法思路:
算法思路
根据题目要求,首先可以想到的是暴力枚举算法(遇事不决,暴力解决),遍历穷举出所有的连续子数组,寻找满足要求的子数组,最终就找到了最小的连续子数组:
class Solution {
public:
    int minSubArrayLen(int s, vector<int>& nums) {
    	//暴力解法
        int n = nums.size();
        if (n == 0) {
            return 0;
        }
        //默认为最大值
        int ans = INT_MAX;
        //开始遍历
        for (int i = 0; i < n; i++) {
        //重置sum值
            int sum = 0;
            //判断子数组是否满足
            for (int j = i; j < n; j++) {
                sum += nums[j];
                if (sum >= s) {
                //满足就更新结果
                    ans = min(ans, j - i + 1);
                    break;
                }
            }
        }
        return ans == INT_MAX ? 0 : ans;
    }
};
 
这样暴力的算法的时间复杂度是O(n^2),我们看看可不可以进行优化:
 来看图解(来着力扣官方)
这样就模拟了滑动窗口:
 做法:将右端元素划⼊窗⼝中,统计出此时窗⼝内元素的和:
- 如果窗⼝内元素之和⼤于等于 target :更新结果,并且将左端元素划出去的同时继续判
断是否满⾜条件并更新结果(因为左端元素可能很⼩,划出去之后依旧满⾜条件) - 如果窗⼝内元素之和不满⾜条件: right++ ,另下⼀个元素进⼊窗⼝。
 
class Solution {
public:
    int minSubArrayLen(int target, vector<int>& nums) {
        int left = 0,right = 0;
        //设置为最大值 保证没有满足的子数组时可以判断
        int len = INT_MAX;
        int sum = 0;
        sum += nums[left];
        while(left < nums.size() && right < nums.size()){
			//
            if(sum < target ){
                right++;
                if(right < nums.size())
                    sum += nums[right];
            }
            while (sum >= target){
                len = min (right - left + 1 , len) ;
                sum -= nums[left];
                left++;
            }
        }
        return len == INT_MAX ? 0:len;
    }
};
 
这样大大提高了算法的效率!!!
 为何滑动窗⼝可以解决问题,并且时间复杂度更低?
- 这个窗⼝寻找的是:以当前窗⼝最左侧元素(记为 left1 )为基准,符合条件的情况。也就是在这道题中,从 left1 开始,满⾜区间和 sum >= target 时的最右侧(记为right1 )能到哪⾥。
 - 我们既然已经找到从 left1 开始的最优的区间,那么就可以⼤胆舍去 left1 。但是如果继续像⽅法⼀⼀样,重新开始统计第⼆个元素( left2 )往后的和,势必会有⼤量重复的计算(因为我们在求第⼀段区间的时候,已经算出很多元素的和了,这些和是可以在计算下次区间和的时候⽤上的)。
 - 此时, rigth1 的作⽤就体现出来了,我们只需将 left1 这个值从sum 中剔除。从right1 这个元素开始,往后找满⾜ left2 元素的区间(此时right1 也有可能是满⾜的,因为 left1 可能很⼩。 sum 剔除掉 left1 之后,依旧满⾜⼤于等于target )。这样我们就能省掉⼤量重复的计算。
 
这样我们不仅能解决问题,⽽且效率也会⼤⼤提升
继续我们来看下一题
Leetcode 3. 无重复字符的最长子串
题目描述

 描述也是十分简单奥,我们接着来看如何解决
算法思路
首先想到的还是暴力枚举啊,我们可以借助哈希表来确定是否重复。
 枚举过程中就会发现左右指针移动方向相同,所以可以进行滑动窗口
- 入窗口(右指针移动)
 - 判断(判断是否需要移动左指针)
 - 出窗口
 - 更新结果
 
class Solution {
public:
    int lengthOfLongestSubstring(string s) {
        int len = 0;
        int n = s.size();
        //使用哈希进行判断是否重复
        int hash[128] = {0};
        int ret = 0;
        for(int left = 0,right = 0; right < n; right++){
       		//进入窗口
            hash[s[right]]++;
			//判断
            while(hash[s[right]] > 1){
            	//出窗口
                hash[s[left]]--;
                left++;
                len--;
            }
      		//更新结果
            len++;
            ret = max(len,ret);
        }
        return ret;
    }
};
 
这样就完美解决。
 其实滑动窗口都是可以套用上面的模版的,不信?来看下一题
Leetcode 1004. 最大连续1的个数 III
题目描述

 题目描述依然简单奥,只是判断条件发生了改变,我们需要来定义一个数字来比较是否满足少于k
算法思路
依旧是:
- 入窗口(右指针移动)
 - 判断(判断是否需要移动左指针)
 - 出窗口
 - 更新结果
 
class Solution {
public:
    int longestOnes(vector<int>& nums, int k) {
        int tmp = 0,left = 0,right = 0,n = nums.size();
        int ret = 0;
        while(right < n){
            if(nums[right] == 0) {
                tmp++;    
            }
            while(tmp > k){
                if(nums[left] == 0) tmp--;
                left++;
            }
            ret = max(ret,right - left + 1);
            right++;
        }
        return ret;
    }
};
 
这样就成功完成解题!!!
总结
滑动窗口问题是可以通过模版来解决:
- 入窗口(右指针移动)
 - 判断(按题分析判断是否需要移动左指针)
 - 出窗口
 - 更新结果
 
这样基本滑动窗口都可以解决,但重要的是理解滑动窗口的思路是如何得到的,是如何从暴力算法优化出来的。
送给大家一句话:
那脑袋里的智慧,就像打火石里的火花一样,不去打它是不肯出来的。——莎士比亚













![绝地求生:[更新周报] 3/20 不停机更新:商城无上新、23号七周年HOT TIME!](https://img-blog.csdnimg.cn/direct/bdf5c3af28954ff5be15dfd350e68e68.png)





