滑动 窗口的最大值

首先是问题描述

我的思路如下:

1:用一个递减双端队列(头尾都可以crud的队列)存放滑动窗口中的值的索引,所以每次都是队头是最大值。

2:这个思路的难点就在于这个双端队列的维持,窗口每次移动都要对队列操作,我称这种操作为:去头,去尾,添值

        去头:双端队列的队头对应窗口中的最大值,但这次窗口移动后,可能已经越过了这个最大值,这个队头就不应该用了

        去尾:给窗口新增值在这个队列中找到它该存在的位置,那些比它老,还比它小的值,窗口再怎么移动也不可能成为最大值,所以直接从尾部去掉就好.

        添值:上述步骤后,在队列尾部添加这个新增值的索引就好.

3:显然时间复杂度o(n),空间复杂度o(n)

如果有视频的话可能好理解些:https://leetcode-cn.com/problems/sliding-window-maximum/solution/shi-pin-jie-xi-shuang-duan-dui-lie-hua-dong-chuang/

具体实现思路:

1:创建双端队列,java的话可以用LinkedList,本来就继承了Queue,而且还是双向链表,也确实是以双端队列的思路实现的

2:将双端队列的维持写成方法

3:队列初始化

4:继续剩余的窗口滑动

 

代码:

class Solution {
    //存的是索引
    LinkedList<Integer> queue=new LinkedList<Integer>();
    int[] nums;
    public int[] maxSlidingWindow(int[] nums, int k) {
        //判空
        if(nums==null||nums.length==0)return new int[0];
        //new结果
        int[] res=new int[nums.length-k+1];
        //
        this.nums=nums;
        
        //初始化双端队列
        for(int i=0;i<k;i++){
            keep_queue(i,k);
        }
        res[0]=nums[queue.getFirst()];

        
        //移动滑动窗口
        for(int x=1,y=k;y<nums.length;x++,y++){
            //插入双端队列
            keep_queue(y,k);
            //输出最大值
            res[x]=nums[queue.getFirst()];
        }

        return res;
    }


    //队列维持方法
    void keep_queue(int index,int k){
        //去头
        if(queue.size()==k||(queue.size()!=0&&queue.getFirst()<=index-k))queue.poll();

        //去尾
        while(queue.size()!=0){
                 if(nums[queue.getLast()]<=nums[index]){
                     queue.removeLast();
                 }else{
                     break;
                 }
             }
        //添值
        queue.add(index);
    }
}

 

 

最后再琢磨琢磨大佬们的代码:

class Solution {
    public int[] maxSlidingWindow(int[] nums, int k) {
        int[] re=new int[nums.length-k+1];
        int loc=0;
        int value=nums[0];
        for(int i=1;i<k;i++)
        {
            if(nums[i]>value)
            {
                value=nums[i];
                loc=i;
            }
        }
        re[0]=value;
        int index=1;
        for(int i=k;i<nums.length;i++)
        {
            if(nums[i]>value)
            {
                value=nums[i];
                loc=i;
                re[index]=value;
                index++;
            }
            else
            {
                if(loc+k>i)
                {
                    re[index]=value;
                    index++;
                }
                else
                {
                    value=nums[i-k+1];
                    loc=i-k+1;
                    for(int j=i-k+2;j<=i;j++)
                    {
                        if(nums[j]>value)
                        {
                            value=nums[j];
                            loc=j;
                        }
                    }
                    re[index]=value;
                    index++;
                }
            }
        }
        return re;
    }

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值