面试常见题型之【滑动窗口】解题模板与示例

概述

算法面试过程中,经常会遇到求解满足某种条件的子串问题,对于这种类型的题,一般可以使用双指针或滑动窗口解答,滑动窗口问题可以认为是一种特殊的双指针。

什么是滑动窗口

在学习计算机网络时,在TCP协议中,为了进行拥塞控制,提出使用滑动窗口进行优化。

滑动窗口,顾名思义是使用一个大小可变的窗口,通过控制窗口左右两端移动的方向和移动步调,来达到找出要查找子序列的目的。左右两端点一般是向前滑动,可以是右端固定时,左端向前滑动;或者左端固定时,右端向前滑动。

滑动窗口法,可以用来解决一些查找满足一定条件的连续区间的性质的问题。由于区间连续,因此当区间发生变化时,可以通过旧有的计算结果对搜索空间进行剪枝,这样便减少了重复计算,降低了时间复杂度。

使用场景

滑动窗口法常用于求解满足某种条件的某段连续区间的最短或最长子序列(一般为子数组、子字符串等),如:

1)最小摘要
2)和大于给定目标值的最短子序列
3)无重复字符的最长子串
4)有k个不同字符的子串
滑动窗口的重要性质是:窗口的左边界和右边界永远只能向右移动,而不能向左移动。这是为了保证滑动窗口的时间复杂度是 O(n)O(n)。如果左右边界向左移动的话,这叫做“回溯”,算法的时间复杂度就可能不止 O(n)O(n)

https://leetcode-cn.com/problems/he-wei-sde-lian-xu-zheng-shu-xu-lie-lcof/solution/shi-yao-shi-hua-dong-

算法思想

滑动窗口问题可以想象成队列,一端在push元素,另一端在pop元素,如下所示:

假设有数组[a b c d e f g h]
一个大小为3的滑动窗口在其上滑动,则有:

[a b c]
  [b c d]
    [c d e]
      [d e f]
        [e f g]
          [f g h]

算法解题步骤如下:

1、声明左右两个指针left和right,初始时都指向起始位置 left = right = 0。
2、满足不了条件是, right 指针不停地后移以扩大窗口 [left, right]接近目标,直到窗口中的序列符合要求。
3、找到一个符合要求的子序列时,停止移动 right的值,转而不断移动左端 left 指针以缩小窗口 [left, right],直到窗口中的序列不再符合要求。同时,每次增加 left前,都要更新一轮结果。
4、重复第 2 和第 3 步,直到 right 到达序列的尽头。

第 2 步相当于在寻找一个可行解,然后第 3 步在优化这个可行解,最终找到最优解。左右指针轮流前进,窗口大小增增减减,窗口不断向右滑动。

算法模板

如何使用滑动窗口问题,需要思考如下两个问题:
第一个问题,窗口何时扩大,何时缩小?
第二个问题,如何移动窗口左右两端,以找到全部的解?

下面以在int类型的数组arr中,查找和为target的子序列为例,总结滑动窗口的做题模板如下:

func template(arr []int, target int) {
	var (
		// 子序列
		path []int 
		// 当前子序列的和
		sum int 
	)
 	
 	# 初始化滑动窗口两端(根据具体情况,对左右边界赋初始值)
    left, right := 0, 0
   
	// 循环遍历
    for left < len(arr) {
    	// 没找到符合条件的子序列,根据情况后移左右端指针位置,以扩大或缩小窗口范围查找出满足条件的子序列
		if sum < target {
            sum += arr[right]
            right++
        } else if sum > target {
            // 当前和大于目标值,通过缩小窗口减少总和,左边界后移
            sum -= arr[left]
            left++
        } else {
            // 找到和为目标值的子序列,放入新开辟的path切片中
            path := make([]int, 0)
            // 将[left, right)范围内的子序列放入path切片中
            for i := left; i < right; i++ {
                path = append(path, arr[i])
            }
            // 找到一个可行解,更新结果值
            res = append(res, path)

            // 找到一个符合条件的子序列后,查找下一个子序列时,从和中去除最左侧元素,
            // 从左侧元素的下一个位置开始再查找符合条件的下一个子序列,因此先加left,然后left再后移
            sum -= arr[left]
            left++
        }
	}
}
   

模板只是一个解题思路,具体的题目可能需要具体分析,但是大体框架是不变的。

题目示例

剑指 Offer 57 - II. 和为s的连续正数序列

题目难度:简单
题目链接:https://leetcode-cn.com/problems/he-wei-sde-lian-xu-zheng-shu-xu-lie-lcof/

题目描述

输入一个正整数 target ,输出所有和为 target 的连续正整数序列(至少含有两个数)。
序列内的数字由小到大排列,不同序列按照首个数字从小到大排列。

示例 1:
输入:target = 9
输出:[[2,3,4],[4,5]]

示例 2:
输入:target = 15
输出:[[1,2,3,4,5],[4,5,6],[7,8]]

限制:
1 <= target <= 10^5

算法实现

  • 使用left、right两个指针分别指向当前子序列起始位置,并累积当前窗口范围内的和。
  • 如何窗口的和小于 target 的时候,窗口的和需要增加,可以通过扩大窗口的范围增加,因此累加当前右边界对应元素后,窗口的右边界向右移动。
  • 当窗口的和大于 target 的时候,窗口的和需要减少,通过缩小窗口以较少和的值,因此在减去当前左侧边界对应位置的元素后,窗口的左边界向右移动
  • 当窗口的和恰好等于 target 的时候,需要将当前窗口为 [i, j)范围内的子序列添加到结果集中。
  • 找到一个 left 开头的子序列后,接下来查找下一个满足条件的子序列。可以从left之后的序列查找,因此从和sum中减去left左边界对应元素的值更新当前子序列的和之后,从left+1 开头的序列查找,所以窗口的左边界要向右移动。
// 使用滑动窗口解决和为target的连续子序列问题
func findContinuousSequence(target int) [][]int {
    var (
        // res为满足条件的子序列集合
        res [][]int 
        // sum为当前子序列的和
        sum int 
    )

    // left、right为滑动窗口的左右边界
    left, right := 1, 1
    // 由于子序列至少包含2个元素,因此左边界小于等于目标值的一半
    for left <= target/2 {
        // 当前和小于目标值,通过扩大窗口范围增加,右边界后移
        if sum < target {
            sum += right
            right++
        } else if sum > target {
            // 当前和大于目标值,通过缩小窗口减少总和,左边界后移
            sum -= left
            left++
        } else {
            // 找到和为目标值的子序列,放入新开辟的path切片中
            path := make([]int, 0)
            // 将[left, right)范围内的子序列放入path切片中
            for i := left; i < right; i++ {
                path = append(path, i)
            }
            // 找到一个可行解,更新结果值
            res = append(res, path)

            // 找到一个符合条件的子序列后,查找下一个子序列时,从和中去除最左侧元素,
            // 从左侧元素的下一个位置开始再查找符合条件的下一个子序列,因此先加left,然后left再后移
            sum -= left
            left++
        }
    }
    
    return res 
}

参考

滑动窗口详解
滑动窗口法模板
Leetcode刷题总结之滑动窗口法(尺取法)
算法与数据结构(一):滑动窗口法总结
什么是滑动窗口,以及如何用滑动窗口解这道题

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
1. 二分法 5 1.1. 什么是二分查找 5 1.2. 如何识别二分法 5 1.3. 二分法模板 6 1.3.1. 模板一 6 1.3.1.1. 模板代码 6 1.3.1.2. 关键属性 7 1.3.1.3. 语法说明 7 1.3.1.4. Lc69:x的平方根 8 1.3.1.5. Lc374:猜数大小 9 1.3.1.6. Lc33:搜索旋转数组 11 1.3.2. 模板二 13 1.3.2.1. 模板代码 13 1.3.2.2. 关键属性 14 1.3.2.3. 语法说明 14 1.3.2.4. Lc278:第一个错误版本 14 1.3.2.5. Lc162:寻找峰值 16 1.3.2.6. Lc153:寻找旋转排序数组最小值 19 1.3.2.7. Lc154:寻找旋转排序数组最小值II 20 1.3.3. 模板三 22 1.3.3.1. 模板代码 22 1.3.3.2. 关键属性 23 1.3.3.3. 语法说明 23 1.3.3.4. LC-34:在排序数组中查找元素的第一个和最后一个 23 1.3.3.5. LC-658:找到K个最接近的元素 25 1.3.4. 小结 28 1.4. LeetCode中二分查找题目 29 2. 双指针 30 2.1. 快慢指针 31 2.1.1. 什么是快慢指针 31 2.1.2. 快慢指针模板 31 2.1.3. 快慢指针相关题目 32 2.1.3.1. LC-141:链表是否有环 32 2.1.3.2. LC-142:环形链表入口 34 2.1.3.3. LC-876:链表的中间节点 37 2.1.3.4. LC-287:寻找重复数 40 2.2. 滑动窗口 43 2.2.1. 什么是滑动窗口 43 2.1.4. 常见题型 44 2.1.5. 注意事项 45 2.1.6. 滑动窗口模板 45 2.1.7. 滑动窗口相关题目 46 2.1.7.1. LC-3:无重复字符的最长子串 47 2.1.7.2. LC-76:最小覆盖子串 49 2.1.7.3. LC-209:长度最小的子数组 54 2.1.7.4. LC-239:滑动窗口最大值 57 2.1.7.5. LC-395:至少有K个重复字符的最长子串 60 2.1.7.6. LC-567:字符串排列 62 2.1.7.7. LC-904:水果成篮 64 2.1.7.8. LC-424:替换后的最长重复字符 66 2.1.7.9. LC-713:乘积小于K的子数组 67 2.1.7.10. LC-992:K个不同整数的子数组 70 2.3. 左右指针 73 2.3.1. 模板 73 2.3.2. 相关题目 73 2.3.2.1. LC-76:删除倒数第N个节点 74 2.3.2.2. LC-61:旋转链表 76 2.3.2.3. LC-80:删除有序数组中的重复项 79 2.3.2.4. LC-86:分割链表 80 2.3.2.5. LC-438:找到字符串中所有字母的异位词 82 3. 模板 85 2.3.2.6. LC-76:删除倒数第N个节点 85

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

love666666shen

谢谢您的鼓励!

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值