题目链接:hdu 3410 Passing the Message
题意:
说那么多,其实就是对于每个a[i],让你找他的从左边(右边)开始找a[j]<a[i]并且a[j]=max(a[j])(k+1<j<i),a[k]>a[i]。
题解:
从左往右维护一个递减的单调队列,每次都从尾巴开始把比a[i]的踢掉,最后踢的那个就是答案。
右边同理。
#include<bits/stdc++.h>
#define F(i,a,b) for(int i=a;i<=b;i++)
using namespace std; const int N=5e4+;
int t,n,a[N],Q[N],ans1[N],ans2[N],head,tail,ic; int main()
{
scanf("%d",&t);
while(t--)
{
scanf("%d",&n);
F(i,,n)scanf("%d",a+i);
head=,tail=;
F(i,,n)
{
int flag=;
while(head<=tail&&a[Q[tail]]<a[i])tail--,flag=;
if(flag)ans1[i]=Q[tail+];else ans1[i]=;
Q[++tail]=i;
}
head=,tail=;
for(int i=n;i>;i--)
{
int flag=;
while(head<=tail&&a[Q[tail]]<a[i])tail--,flag=;
if(flag)ans2[i]=Q[tail+];else ans2[i]=;
Q[++tail]=i;
}
printf("Case %d:\n",++ic);
F(i,,n)printf("%d %d\n",ans1[i],ans2[i]);
}
return ;
}