题意:
此题为一道交互题;
给出一个长度为n的序列,求长度为k的字典序最大的子序列;
n<=15000000,k<=1000000;
题解:
这道题建议去Poi官网提交;
BZ不支持交互把这题弄成了sb题;
在官网的内存限制是32MB,并不能存下长度为n的数组;
所以这道题是一个伪在线题目;
首先最基础的离线问题怎么解?
从[1,n-k+1]选最大的做第一个,然后从[第一个+1,n-k+2]选第二个...选到n个为止;
这里我们用O(n)的单调栈实现;
但是这是一个在线问题,和离线比要有一些改进;
首先数组开不下n,所以也要限制栈内元素个数,就是元素大于n个把栈顶弹掉;
还有一个隐藏的问题是,可能最后k个元素较大,将栈弹空导致元素不够n个;
这个问题的关键是没给n,数组还是存不下来的;
所以利用伪在线的方法搞,每次读入到i的时候,处理i-k那里的元素入栈;
这样读完停下的时候,再处理最后k个就好了;
代码我给出Poi官网的交互程序吧,波兰人那一坨函数能看?
(不要在意这个是单调队列的样子。。其实它的内心是单调栈= =)
代码:
#include<stdio.h> #include<string.h> #include<algorithm> #include"carclib.h" #define N 1001000 using namespace std; int a[N]; int q[N],st,en; int main() { int n,m,i,j,k,x; k=inicjuj(); for(i=1,n=0;i<=k;i++) { x=wczytaj(); a[++n]=x; } st=1,en=0; while(x=wczytaj()) { a[++n%N]=x; while(st<=en&&q[en%N]<a[(n-k)%N]) en--; while(en-st>k) en--; q[++en%N]=a[(n-k)%N]; } for(i=k;i>=1;i--) { while(st<=en&&q[en%N]<a[(n-i+1)%N]) en--; q[(++en)%N]=a[(n-i+1)%N]; wypisz(q[st++%N]); } return 0; }