9
29
2015
0

4149: [AMPPZ2014]Global Warming

4149: [AMPPZ2014]Global Warming

Time Limit: 20 Sec  Memory Limit: 256 MB
Submit: 112  Solved: 38
[Submit][Status][Discuss]

Description

给定一个序列a[1],a[2],...,a[n]。请从中选出一段连续子序列,使得该区间最小值唯一、最大值也唯一。
输出选出的子序列的长度的最大值以及取到最大值时左端点的最小值。
 

 

Input

第一行包含一个正整数n(1<=n<=500000),表示序列长度。
第二行包含n个正整数,依次表示a[1],a[2],...,a[n](-10^9<=a[i]<=10^9)。
 

 

Output

包含一行两个整数l,k,其中l表示选出的子序列的长度的最大值,k表示取到最大值时左端点的最小值。
 

 

Sample Input

10
8 3 2 5 2 3 4 6 3 6

Sample Output

6 4

HINT

 

选出的子序列为5,2,3,4,6,3,只有唯一的最小值2和唯一的最大值6。


 

 

Source

 

#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
const int N = 500010;
struct data{int x,s,y,next;} g[N],h[N];
struct node{int l,r;} x[N],y[N];
int n,m,a[N],q[N],t,ans,Pos,j,cnth,cntg,v[N<<2],headh[N],headg[N];
int read(){
	int x = 0,f = 1,c = getchar();
	while (c<'0' || c>'9') {if (c=='-') f = -1;c = getchar();}
	while (c>='0' && c<='9') x = x*10+c-'0',c = getchar();
	return x*f;
}
void Inserth(int x,int s,int y){h[++cnth] = (data){x,s,y,headh[x]};headh[x] = cnth;}
void Insertg(int x,int s,int y){g[++cntg] = (data){x,s,y,headg[x]};headg[x] = cntg;}
void Ins(int x,int val){
	int l = 1,r = n,pos = 1;
	while (1){
		v[pos] = max(v[pos],val);
		if (l==r) return;
		int mid = (l+r)>>1;
		if (x<=mid){r = mid;pos = pos<<1;}
		else {l = mid+1;pos = pos<<1|1;}
	}
}
int Query(int pos,int l,int r,int x,int y){
	if (l==x && r==y){return v[pos];}
	int mid = (x+y)>>1;
	if (r<=mid) return Query(pos<<1,l,r,x,mid);
	else if (l>mid) return Query(pos<<1|1,l,r,mid+1,y);
	else return max(Query(pos<<1,l,mid,x,mid),Query(pos<<1|1,mid+1,r,mid+1,y));
}
void work(){
	for (int i=1;i<=n;i++) Insertg(x[i].l,i,x[i].r),Inserth(y[i].l,i,y[i].r);
	for (int x=1;x<=n;x++){
		for (int k=headh[x];k;k=h[k].next) Ins(h[k].s,h[k].y);
		for (int k=headg[x];k;k=g[k].next){
			j = Query(1,x,g[k].y,1,n);
			if (j<g[k].s) continue;
			if ((t=min(j,g[k].y)-x+1)>ans){ans = t;Pos = x;}
			else if (t==ans){Pos = min(Pos,x);}
		}
	}
}
int main(){
	n = read();
	for (int i=1;i<=n;i++) a[i] = read();
	q[0] = 0;t = 0;
	for (int i=1;i<=n;i++){while (t && a[q[t]]>a[i]) t--;x[i].l = q[t]+1;q[++t] = i;}
	q[0] = n+1;t = 0;
	for (int i=n;i;i--){while (t && a[q[t]]>a[i]) t--;x[i].r = q[t]-1;q[++t] = i;}
	q[0] = 0;t = 0;
	for (int i=1;i<=n;i++){while (t && a[q[t]]<a[i]) t--;y[i].l = q[t]+1;q[++t] = i;}
	q[0] = n+1;t = 0;
	for (int i=n;i;i--){while (t && a[q[t]]<a[i]) t--;y[i].r = q[t]-1;q[++t] = i;}
	work();
	cntg = cnth = 0;memset(v,0,sizeof(v));
	for (int i=1;i<=n;i++) headh[i] = headg[i] = 0;
	for (int i=1;i<=n;i++) swap(x[i],y[i]);
	work();
	printf("%d %d",ans,Pos);
}
Category: BZOJ题解 | Tags: | Read Count: 494

登录 *


loading captcha image...
(输入验证码)
or Ctrl+Enter

Host by is-Programmer.com | Power by Chito 1.3.3 beta | Theme: Aeros 2.0 by TheBuckmaker.com