hdu3518 Boring Counting[后缀排序]

裸的统计不同的重复出现子串(不重叠)种数的题。
多次使用后缀排序要注意小细节。y数组在重复使用时一定要清空,看那个line25 +k就明白了 ,cnt也要清空,为什么就不说了
 1 #include<bits/stdc++.h>
 2 using namespace std;
 3 typedef long long ll;
 4 template<typename T>inline char MAX(T&A,T B){return A<B?(A=B,1):0;}
 5 template<typename T>inline char MIN(T&A,T B){return A>B?(A=B,1):0;}
 6 const int N=1000+7;
 7 char s[N];
 8 int n,m;
 9 ll ans;
10 
11 int sa[N],rk[N],x[N],y[N],cnt[N],h[N],p;
12 inline void build(){
13     memset(cnt,0,sizeof cnt);memset(y,0,sizeof y);//
14     for(register int i=1;i<=n;++i)++cnt[x[i]=s[i]];
15     for(register int i=1;i<=m;++i)cnt[i]+=cnt[i-1];
16     for(register int i=n;i;--i)sa[cnt[x[i]]--]=i;
17     for(register int k=1;k<n;k<<=1,p=0){
18         for(register int i=n-k+1;i<=n;++i)y[++p]=i;
19         for(register int i=1;i<=n;++i)if(sa[i]>k)y[++p]=sa[i]-k;
20         for(register int i=1;i<=m;++i)cnt[i]=0;
21         for(register int i=1;i<=n;++i)++cnt[x[y[i]]];
22         for(register int i=1;i<=m;++i)cnt[i]+=cnt[i-1];
23         for(register int i=n;i;--i)sa[cnt[x[y[i]]]--]=y[i];
24         swap(x,y);x[sa[1]]=p=1;
25         for(register int i=2;i<=n;++i)x[sa[i]]=y[sa[i-1]]==y[sa[i]]&&y[sa[i-1]+k]==y[sa[i]+k]?p:++p;
26         if(p==n)break; m=p;
27     }p=0;
28 //  for(register int i=1;i<=n;++i)printf("%d %d\n",i,x[i]); 
29     for(register int i=1;i<=n;h[x[i]]=p,p?--p:1,++i)while(s[i+p]==s[sa[x[i]-1]+p]&&++p);
30 }
31 
32 inline void search(int k){
33     int l=sa[1],r=sa[1];
34     for(register int i=2;i<=n;++i){
35         if(h[i]<k){r-l>=k?++ans:1;l=r=sa[i];}
36         MIN(l,sa[i]),MAX(r,sa[i]);
37     }
38     r-l>=k?++ans:1;
39 }
40 
41 int main(){
42     while(~scanf("%s",s+1)){
43         if(s[1]=='#')break;
44         n=strlen(s+1),m=127,ans=0,p=0,build();
45         for(register int i=1;i<=n/2;++i)search(i);
46         printf("%lld\n",ans);
47     }
48     return 0;
49 }

 



posted @ 2019-01-31 00:05  Ametsuji_akiya  阅读(130)  评论(0编辑  收藏  举报