zoukankan      html  css  js  c++  java
  • 【后缀数组】uoj#35. 后缀排序

    模板

    #include<cstdio>
    #include<algorithm>
    #include<cstring>
    using namespace std;
    #define N 100001
    int sa[N],t[N],t2[N],tong[N],n;
    char s[N];
    inline bool cmp(int *y,int i,int k)
    {
    	return (y[sa[i-1]]==y[sa[i]])&&((sa[i-1]+k>=n?-1:y[sa[i-1]+k])==(sa[i]+k>=n?-1:y[sa[i]+k]));
    }
    //构造字符串s的后缀数组,每个字符值必须为0~m-1,字符串下标为0~n-1 
    void build_sa(int range)
    {
    	int *x=t,*y=t2;
    	//基数排序
    	memset(tong,0,sizeof(int)*range);//清空桶 
    	for(int i=0;i<n;++i) tong[x[i]=s[i]]++;//把s拷贝到x中,之后插入桶
    	for(int i=1;i<range;++i) tong[i]+=tong[i-1];//将桶处理成前缀和
    	for(int i=n-1;i>=0;--i) sa[--tong[x[i]]]=i;
    	for(int k=1;k<=n;k<<=1)
    	  {
    	  	int p=0;
    	  	//直接利用sa数组排序第二关键字 
    	  	for(int i=n-k;i<n;++i) y[p++]=i;
    	  	for(int i=0;i<n;++i) if(sa[i]>=k) y[p++]=sa[i]-k;
    	  	//基数排序第一关键字
    	  	memset(tong,0,sizeof(int)*range);
    		for(int i=0;i<n;++i) tong[x[y[i]]]++;
    		for(int i=0;i<range;++i) tong[i]+=tong[i-1];
    		for(int i=n-1;i>=0;--i) sa[--tong[x[y[i]]]]=y[i];
    		//根据sa和y数组计算新的x数组 
    		swap(x,y);
    		p=1; x[sa[0]]=0;
    		for(int i=1;i<n;++i) x[sa[i]]= cmp(y,i,k) ? p-1 : p++;
    		if(p>=n) break;
    		range=p;
    	  }
    }
    int rank[N],lcp[N];
    void get_lcp()
    {
    	int k=0;
    	for(int i=0;i<n;++i) rank[sa[i]]=i;
    	for(int i=0;i<n;++i) if(rank[i])
    	  {
    	  	if(k) --k;
    	  	int j=sa[rank[i]-1];
    	  	while(s[i+k]==s[j+k]) ++k;
    	  	lcp[rank[i]]=k;
    	  }
    }
    int main()
    {
    	scanf("%s",s);
    	n=strlen(s);
    	build_sa('z'+1);
    	for(int i=0;i<n;++i) printf("%d ",sa[i]+1);
    	puts("");
    	if(n>1) get_lcp();
    	for(int i=1;i<n;++i) printf("%d ",lcp[i]);
    	return 0;
    }
  • 相关阅读:
    有向无环图
    2016
    Permutation Descent Counts(递推)
    Rikka with Subset
    hihoCoder 1549 或运算和
    Young Maids
    1925: [Sdoi2010]地精部落
    Problem H. Hotel in Ves Lagos
    改变presentModalView大小
    no such file to load -- bundler/setup
  • 原文地址:https://www.cnblogs.com/autsky-jadek/p/4458041.html
Copyright © 2011-2022 走看看