赞
踩
算法需要多刷题积累经验,所以我行文重心在于分析解题思路,理论知识部分会相对简略一些
滑动窗口属于双指针,这两个指针是同向前行
,它们所夹的区间就称为“窗口”
啥时候用滑动窗口?
如何使用?(步骤)
移动 right
让元素进窗口2和3需要放在一个循环
里面,这样才可以让窗口不断往前走
思路:
[left,right]
区间中的元素和大于target,此时得到区间长度为right-left+1
全是正数
,此时right继续走的话,区间中元素总和肯定还是比target大,但是此时区间的长度肯定不是最小长度,这样就做了无用功left
向前走,缩小区间,将新区间的元素和与 target 比较,若大于 target,则记录此时区间长度,并让 left 继续向前走;反之则让 right 往前走上面的思路其实就是从暴力枚举逐步过渡到滑动窗口,整个过程下来,我们不难发现:砍掉暴力枚举中那些无用功,也就得到滑动窗口的思路了
代码如下:
class Solution { public int minSubArrayLen(int target, int[] nums) { int left = 0,right = 0; //区间为左闭右闭 int sum = 0; //区间中元素总和 int minLen = nums.length+1; //最小窗口的长度,一开始初始化为一个比较大的数,不然下面使用取小函数可能没法正确更新minLen for(;right < nums.length;right++) { sum += nums[right]; while(sum >= target) { minLen = Math.min(minLen,right-left+1); //更新 minLen sum -= nums[left]; left++; } } return minLen == nums.length+1 ? 0 : minLen; } }
注意:用取小函数来更新 minLen,可以简化代码(相比于使用条件语句判断大小),同理,可以用取大函数来更新最大值
时间复杂度:O(N),最坏情况下左右指针都要走完整个数组
比如数组大小为5,前四个元素都为1,最后一个为10000,target为10
空间复杂度:O(1)
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。