bzoj4970 [ioi2004]empodia 障碍段

Description 

古数学及哲学家毕氏相信自然之本质为数学。现代生物学家研究生物数列(biosequences)。 生物数数为满足下列条件之 M 个整数所成的数数:
1: 包含从 0, 1, …, 到 M - 1 的所有数字
2: 起始数字为 0, 最后一个数字为 M - 1
?2:数列中 E+1 不可以紧接在 E 之后
生物数数的连续子数列称为数段(segments)。如果一个数段的起点为该数段最小的数字, 终点为该数段最大的数字且与起点不是同一个数字,且介于这两个数字之间所有的整数都出现在这个数段中, 则称这个数段为框段(framed interval),如果框段中并不包含题名小的框段,则称之为障碍段(empodio)。以(0,3,5,4,6,2,1,7)这个生物数列为例。 整个生物数?是一个框段, 可是它包含了另外一框段 (3,5,4,6) ,因此该生物数列是障碍段。而框段 (3,5,4,6) 并不包含任何更短的框段所以它是一个障碍段,而且是此生物数列中唯一的障碍段。请写一个程序, 在输入生物数列后, 输出所有的障碍段 (empodia 为 empodio的复数形)。 

Input

第一行为单一整数M,代表生物数列的长度。 
生物数列中的数字依序出现在接下来的 M 行,每一行有一个整数
M≤1100000 

Output

第一行为一整数H,代表该生物数列中的障碍段的个数。
接下来的 H 行,将每一个障碍段,依照起点在原输入生物数列中出现的顺序,依序输出。
每行以2个整数A 与 B 代表一个障碍段并以一个空白分开
原输入生物数列第 A 个元素为该障碍段之起点,而第 B 个元素为该障碍段之终点

Sample Input

8
0
3
5
4
6

Sample Output

1
2 5
 
正解:单调栈。
联赛模拟考了这道题。
讲道理这题并不难,但是全场都在卡$t2$常数,所以写这题的时候还剩十分钟,匆匆暴力了事。
我们考虑一下一个合法区间的充要条件。
首先$a[i]-i==a[j]-j$,这个还是比较显然的。
然后我们记录$lim1[i]$表示以$a[i]$作为最小值,$i$能延伸到的最右端点。
$lim2[i]$表示以$a[i]$作为最大值,$i$能延伸到的最左端点。
显然,$lim1[l]>=r,lim2[r]<=l$,然后这两个可以用单调栈求出来。
然后我们把$a[i]-i$相等的元素分为一组,每次维护一个$lim1[i]$递减的单调栈。
先把栈顶不合法的弹掉,然后我们就可以直接判断当前元素和栈顶是否可以组成一个合法解。
得到所有可行解以后,我们还要去掉包含关系的区间,于是对所有区间按照左端点从小到大排序,再做一遍单调栈。
好像最后一步有更简单的做法,不过懒得改了。
 
 1 #include <bits/stdc++.h>
 2 #define il inline
 3 #define RG register
 4 #define ll long long
 5 #define N (2000005)
 6 
 7 using namespace std;
 8 
 9 struct data{ int x,y; }ans[N];
10 struct edge{ int nt,to; }g[N];
11 
12 int head[N],lim1[N],lim2[N],st[N],a[N],p[N],n,num,tot,top,Ans;
13 
14 il int gi(){
15   RG int x=0,q=1; RG char ch=getchar();
16   while ((ch<'0' || ch>'9') && ch!='-') ch=getchar();
17   if (ch=='-') q=-1,ch=getchar();
18   while (ch>='0' && ch<='9') x=x*10+ch-48,ch=getchar();
19   return q*x;
20 }
21 
22 il void insert(RG int from,RG int to){
23   g[++num]=(edge){head[from],to},head[from]=num; return;
24 }
25 
26 il void solve(){
27   for (RG int i=1;i<=tot;++i){
28     while (top && lim1[st[top]]<p[i]) --top;
29     if (top && lim2[p[i]]<=st[top]) ans[++Ans]=(data){st[top],p[i]};
30     while (top && lim1[st[top]]<=lim1[p[i]]) --top; st[++top]=p[i];
31   }
32   return;
33 }
34 
35 il int cmp(const data &a,const data &b){
36   if (a.x==b.x) return a.y<b.y; return a.x<b.x;
37 }
38 
39 int main(){
40 #ifndef ONLINE_JUDGE
41   freopen("empodia.in","r",stdin);
42   freopen("empodia.out","w",stdout);
43 #endif
44   n=gi(); for (RG int i=1;i<=n;++i) a[i]=gi()+1;
45   for (RG int i=n;i;--i) insert(n+a[i]-i,i);
46   for (RG int i=1;i<=n;++i){
47     while (top && a[st[top]]<a[i]) --top;
48     lim2[i]=st[top]+1,st[++top]=i;
49   }
50   st[top=0]=n+1;
51   for (RG int i=n;i;--i){
52     while (top && a[st[top]]>a[i]) --top;
53     lim1[i]=st[top]-1,st[++top]=i;
54   }
55   for (RG int i=0,j;i<=(n<<1);++i){
56     for (tot=top=0,j=head[i];j;j=g[j].nt) p[++tot]=g[j].to;
57     if (tot>=2) solve();
58   }
59   sort(ans+1,ans+Ans+1,cmp),top=0;
60   for (RG int i=1;i<=Ans;++i){
61     while (top && ans[st[top]].y>=ans[i].y) --top;
62     if (top && ans[st[top]].x==ans[i].x) continue; st[++top]=i;
63   }
64   cout<<top<<endl;
65   for (RG int i=1;i<=top;++i) printf("%d %d\n",ans[st[i]].x,ans[st[i]].y);
66   return 0;
67 }

 

posted @ 2017-11-01 22:29  wfj_2048  阅读(340)  评论(0编辑  收藏  举报