239 滑动窗口最大值
题目:
给定一个数组 nums,有一个大小为 k 的滑动窗口从数组的最左侧移动到数组的最右侧。你只可以看到在滑动窗口 k 内的数字。滑动窗口每次只向右移动一位。
返回滑动窗口最大值。
示例:
输入: nums = [1,3,-1,-3,5,3,6,7], 和 k = 3
输出: [3,3,5,5,6,7]
解释:
滑动窗口的位置 最大值
[1 3 -1] -3 5 3 6 7 3
1 [3 -1 -3] 5 3 6 7 3
1 3 [-1 -3 5] 3 6 7 5
1 3 -1 [-3 5 3] 6 7 5
1 3 -1 -3 [5 3 6] 7 6
1 3 -1 -3 5 [3 6 7] 7
方法:双端队列法
思路:
1.创建一个双端队列,记下滑动窗口里的数的下标
2.为保证队列最左端的数是最大数的下标,如果新输入的下标的值大于最左端下标的值,就全部清空,如果不大于则移动下标继续判断,小于则除去
3.当滑窗的大小确定,每次移动都保证滑窗两边都在增加
4.当滑窗大小确定,就可以开始将最左端的值代表的数加入result数组中
代码:
public int[] maxSlidingWindow(int[] nums, int k) {
if(nums==null||nums.length<2) return nums;
//双端队列
LinkedList<Integer> list = new LinkedList<>();
//结果数组
int[] result = new int[nums.length-k+1];
for (int i = 0; i < nums.length ; i++) {
//如果新来的数大于前面的数,则弹出前面的数
while(!list.isEmpty()&&nums[list.peekLast()]<=nums[i]){
list.pollLast();
}
list.addLast(i);
if(list.peek()<=i-k){
list.poll();
}
if(i-k+1>=0){
result[i-k+1]=nums[list.peek()];
}
}
return result;
}