1692: [Usaco2007 Dec]队列变换
Description
FJ打算带他的N(1 <= N <= 30,000)头奶牛去参加一年一度的“全美农场主大奖赛”。在这场比赛中,每个参赛者都必须让他的奶牛排成一列,然后领她们从裁判席前依次走过。 今年,竞赛委员会在接受队伍报名时,采用了一种新的登记规则:他们把所有队伍中奶牛名字的首字母取出,按它们对应奶牛在队伍中的次序排成一列(比如说,如果FJ带去的奶牛依次为Bessie、Sylvia、Dora,登记人员就把这支队伍登记为BSD)。登记结束后,组委会将所有队伍的登记名称按字典序升序排列,就得到了他们的出场顺序。 FJ最近有一大堆事情,因此他不打算在这个比赛上浪费过多的时间,也就是说,他想尽可能早地出场。于是,他打算把奶牛们预先设计好的队型重新调整一下。 FJ的调整方法是这样的:每次,他在原来队列的首端或是尾端牵出一头奶牛,把她安排到新队列的尾部,然后对剩余的奶牛队列重复以上的操作,直到所有奶牛都被插到了新的队列里。这样得到的队列,就是FJ拉去登记的最终的奶牛队列。 接下来的事情就交给你了:对于给定的奶牛们的初始位置,计算出按照FJ的调整规则所可能得到的字典序最小的队列。
Input
* 第1行: 一个整数:N
* 第2..N+1行: 第i+1行仅有1个'A'..'Z'中的字母,表示队列中从前往后数第i 头奶牛名字的首字母
Output
* 第1..??行: 输出FJ所能得到的字典序最小的队列。每行(除了最后一行)输 出恰好80个'A'..'Z'中的字母,表示新队列中每头奶牛姓名的首 字母
Sample Input
6
A
C
D
B
C
B
输入说明:
FJ有6头顺次排好队的奶牛:ACDBCB
A
C
D
B
C
B
输入说明:
FJ有6头顺次排好队的奶牛:ACDBCB
Sample Output
ABCBCD
输出说明:
操作数 原队列 新队列
#1 ACDBCB
#2 CDBCB A
#3 CDBC AB
#4 CDB ABC
#5 CD ABCB
#6 D ABCBC
#7 ABCBCD
输出说明:
操作数 原队列 新队列
#1 ACDBCB
#2 CDBCB A
#3 CDBC AB
#4 CDB ABC
#5 CD ABCB
#6 D ABCBC
#7 ABCBCD
第一次写抄后缀数组=,=
之前有道题似乎因为数据比较弱,就用暴力过了,现在用后缀数组来过掉。。
暴力之所以暴力是因为在字符串的头和尾一样的时候就要不断的去寻找里面不一样的字符,而这个时候就可以使用后缀数组优化了。。。
upd:具体做法就是把字符串翻着复制一遍放在尾部,求出sa,然后实际上就是一个比较前缀和后缀的事情了。。。。
1 #include<iostream> 2 #include<cstdlib> 3 #include<cmath> 4 #include<cstring> 5 #include<cstdio> 6 #include<algorithm> 7 #include<string> 8 #include<map> 9 #include<queue> 10 #include<vector> 11 #include<set> 12 #define mod 1000000007 13 #define inf 1000000000 14 #define maxn 60005 15 #define maxm 30005*2 16 #define eps 1e-10 17 #define ll long long 18 #define for0(i,n) for(int i=0;i<(n);i++) 19 #define for1(i,n) for(int i=1;i<(n);i++) 20 #define for2(i,x,y) for(int i=(x);i<=(y);i++) 21 #define for3(i,x,y) for(int i=(x);i>=(y);i--) 22 #define for4(i,x) for(int i=head[x],y=e[i].go;i;i=e[i].next,y=e[i].go) 23 using namespace std; 24 int read(){ 25 int x=0,f=1;char ch=getchar(); 26 while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();} 27 while(ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();} 28 return x*f; 29 } 30 char s[maxn]; 31 int n,m,sa[maxn],t[maxn],t2[maxn],c[maxn],rank[maxn]; 32 void getsa(int m){ 33 int *x=t,*y=t2; 34 for0(i,m)c[i]=0; 35 for0(i,n)c[x[i]=s[i]]++; 36 for1(i,m)c[i]+=c[i-1]; 37 for3(i,n-1,0)sa[--c[x[i]]]=i; 38 for(int k=1;k<=n;k<<=1){ 39 int p=0; 40 for(int i=n-k;i<n;i++)y[p++]=i; 41 for(int i=0;i<n;i++)if(sa[i]>=k)y[p++]=sa[i]-k; 42 for(int i=0;i<m;i++)c[i]=0; 43 for(int i=0;i<n;i++)c[x[y[i]]]++; 44 for(int i=1;i<m;i++)c[i]+=c[i-1]; 45 for(int i=n-1;i>=0;i--)sa[--c[x[y[i]]]]=y[i]; 46 swap(x,y); 47 p=1;x[sa[0]]=0; 48 for(int i=1;i<n;i++){ 49 x[sa[i]]=y[sa[i-1]]==y[sa[i]]&&y[sa[i-1]+k]==y[sa[i]+k]?p-1:p++; 50 } 51 if(p>=n)break; 52 m=p; 53 } 54 } 55 int main(){ 56 freopen("input.txt","r",stdin); 57 freopen("output.txt","w",stdout); 58 n=read(); 59 for(int i=0;i<n;i++)scanf("%s",&s[i]); 60 s[n]='$'; 61 for(int i=0;i<n;i++)s[i+n+1]=s[n-i-1]; 62 m=n;n<<=1;n=n+1; 63 s[n]='\0'; 64 getsa(128); 65 for(int i=0;i<n;i++)rank[sa[i]]=i; 66 int cc=0; 67 int l=0,r=m-1; 68 for(int i=0;i<m;i++){ 69 int a=rank[l],b=rank[m-r+m]; 70 cc++; 71 if(a<b){ 72 putchar(s[l++]); 73 }else putchar(s[r--]); 74 if(cc==80) cc=0,putchar('\n'); 75 } 76 return 0; 77 }