单调队列

91 阅读2分钟

队列通常都是遵守先进先出的规矩,而单调队列是用于维护一个单调递增或递减的序列。它通常用于解决涉及范围查询或滑动窗口问题的问题,例如在数组中找到最小值或最大值。基本操作包括插入、删除和查询,确保队列始终保持单调性。所以单调队列又是变相的双端队列,可以进行队头队尾的增删查

就如滑动窗口来讲解

image.png

通常情况下面对这样的题,我们用暴力法更为思路清晰:从头到尾扫描,每次查询k个数,一共检查O(nk)次,那么暴力法肯定会超时,下面用单调队列实现它的复杂度为O(n);

在本题当中单调队列有以下特征:

  • 队头的元素始终是队列中最小的。根据需要输出队头,但是不一定弹出。
  • 元素只能从队尾进入队列,从队头、队尾都可以弹出。
  • 序列中的每个元素都必须进入队列。例如,x进入队尾时,和原队尾y比较,如果r≤y,就从队尾弹出y;一直到弹出队尾所有比x大的元素,最后r进人队尾。这个人队操作保证了队头元素是队列中最小的。

从以上过程可以看出单调队列有两个重要的操作:删头,去尾

  • 删头: 如果队列的元素脱离了窗口就弹出
  • 去尾: 如果新元素进入队尾时原队尾的存在破坏了都单调性,就弹出原队尾

实现代码如下:

import java.io.*;
import java.time.chrono.MinguoChronology;
import java.util.*;

public class Main {
    static void solve(){
        int n=sc.nextInt();
        int k=sc.nextInt();
        long []a=new long[1000005];
        Deque<Integer>q=new LinkedList<>();
        for (int i = 1; i <=n; i++) {
            a[i]=sc.nextInt();
        }
        for(int i=1;i<=n;i++){
            while (!q.isEmpty()&&a[q.peekLast()]>a[i])q.removeLast();
            q.offerLast(i);
            if(i>=k){
                while(!q.isEmpty()&&q.peekFirst()<=i-k)q.pollFirst();
               out.print(a[q.peekFirst()]+" ");
            }
        }
        out.println();
        Deque<Integer>q1=new LinkedList<>();
        for(int i=1;i<=n;i++){
            while (!q1.isEmpty()&&a[q1.peekLast()]<a[i])q1.removeLast();
            q1.offerLast(i);
            if(i>=k){
                while(!q1.isEmpty()&&q1.peekFirst()<=i-k)q1.pollFirst();
                out.print(a[q1.peekFirst()]+" ");
            }
        }

    }


    public static void main(String[] args) {
        int  T=1;
        while(T-->0){
            solve();
        }
        out.flush();
        out.close();
    }



    static Kattio sc = new Kattio();
    static PrintWriter out = new PrintWriter(new BufferedWriter(new OutputStreamWriter(System.out)));

    static class Kattio {
        static BufferedReader r;
        static StringTokenizer st;

        public Kattio() {
            r = new BufferedReader(new InputStreamReader(System.in));
        }

        public String next() {
            try {
                while (st == null || !st.hasMoreTokens()) {
                    st = new StringTokenizer(r.readLine());
                }
                return st.nextToken();
            } catch (Exception e) {
                return null;
            }
        }

        public int nextInt() {
            char[] str = next().toCharArray();
            int i = 0;
            boolean neg = false;
            if (str[0] == '-') {
                i = 1;
                neg = true;
            }
            int ans = 0;
            for (; i < str.length; i++) ans = ans * 10 + (str[i] - '0');
            return neg ? -ans : ans;
        }

        public long nextLong() {
            char[] str = next().toCharArray();
            int i = 0;
            boolean neg = false;
            if (str[0] == '-') {
                i = 1;
                neg = true;
            }
            long ans = 0;
            for (; i < str.length; i++) ans = ans * 10 + (str[i] - '0');
            return neg ? -ans : ans;
        }

        public double nextDouble() {
            return Double.parseDouble(next());
        }
    }
}