滑动窗口算法详解
算法目的
该算法展示了如何将嵌套for循环在少数问题中转换为单个for循环,从而减小时间复杂度。
一个经典问题
给一组大小为n的整数数组,计算长度为k的子数组和的最大值
该技术可以通过总线上的窗格得到很好的理解,考虑长度为n的窗口和长度为k的窗格。考虑一下,最初窗格处于
最左边,即从左边开始的0个单位。现在,将窗口与大小为n的数组以长度为k的子段的和current_sum相关联。现在,
如果我们在窗口上施加力量,使其向前移动一个单位距离。该窗口将覆盖下一个k个连续元素。
应用滑动窗口技术
- 我们使用线性循环计算n个项中前k个元素的总和,并将其总和存储在变量window_sum中。
- 然后,我们在数组上线性滑动直到窗口右边界达到数组右边界
- 要获得k个元素块的当前总和,只需要从前一个块中前去第一个元素并添加当前块的最后一个元素即可。
这是我们计算从索引0开始的初始窗口总和的初始阶段。在这个阶段,窗口和为6.现在,我们将maximum_sum设置为current_window,即6。
现在,我们用单位索引来滑动我们的窗口。因此,现在它从窗口中丢弃5并将0添加到窗口。因此,我们将得到新的窗口总和,减去5,然后加上0。所以,我们的窗口和现在变成1.现在,我们将比较这个窗口和与maximum_sum。因为它更小,我们不会改变maximum_sum。
同样,现在我们再次用一个单位索引来滑动我们的窗口,并获得新的窗口总和为2.我们再一次检查这个当前窗口总和是否大于maximum_sum,直到现在。有一次,它再小一些,所以我们不改变maximum_sum。
因此,对于上面的数组,我们的maximum_sum是6。
总结
现在,很明显时间复杂性是线性的,因为我们可以看到只有一个循环运行在我们的代码中。因此,我们的时间复杂度是O(n)。
我们可以使用这种技术来查找最大/最小k-子阵列,XOR,乘积,总和等