【BZOJ】【P3172】【Tjoi2013】【单词 】【题解】【SA】

传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=3172

后缀数组模板题

Code:

#include<bits/stdc++.h>
#define fst first
#define sec second
using namespace std;
typedef pair<int,int> pi;
const int maxn=1e6+205;
pi tmp[maxn];
int sa[maxn],rank[maxn],height[maxn],n,len,len2;
char s[maxn];
int sum[maxn],lef[maxn],rig[maxn];
void radix_sort(int n,pi *a,int *rank,int *sa){
	int m=max(n,27);
	memset(sum,0,sizeof(int)*(m+1));
	for(int i=1;i<=n;i++)sum[a[i].sec]++;
	for(int i=1;i<=m;i++)sum[i]+=sum[i-1];
	for(int i=n;i>=1;i--)rank[i]=sum[a[i].sec]--;
	for(int i=1;i<=n;i++)sa[rank[i]]=i;
	memset(sum,0,sizeof(int)*(m+1));
	for(int i=1;i<=n;i++)sum[a[i].fst]++;
	for(int i=1;i<=m;i++)sum[i]+=sum[i-1];
	for(int i=n;i>=1;i--)rank[sa[i]]=sum[a[sa[i]].fst]--;
	for(int i=1;i<=n;i++)sa[rank[i]]=i;rank[sa[1]]=1;
	for(int i=2;i<=n;i++)rank[sa[i]]=rank[sa[i-1]]+(a[sa[i]]!=a[sa[i-1]]);
}
void sa_init(int n){
	for(int i=1;i<=n;i++)tmp[i].fst=s[i]-'a'+1,tmp[i].sec=0;
	radix_sort(n,tmp,rank,sa);int mx=0;
	for(int j=1;j<=n;j<<=1){
		for(int i=1;i<=n;i++)tmp[i].fst=rank[i],tmp[i].sec=i+j<=n?rank[i+j]:0,mx=max(mx,rank[i]);
		if(mx==n)break;radix_sort(n,tmp,rank,sa);
	}for(int i=1,j,k=0;i<=n;i++){
		k-=!!k;j=sa[rank[i]-1];
		while(s[i+k]==s[j+k])k++;
		height[rank[i]]=k;
	}
}
int Qsum(int L,int R){
	int l=1,r=len,ans=0;
	while(l<r){
		int mid=(l+r)>>1;
		int res=0;
		for(int i=L;i<=R;i++){
			if(sa[mid]+i-L>len){res=1;break;}
			if(s[i]<s[sa[mid]+i-L]){res=-1;break;}
			if(s[i]>s[sa[mid]+i-L]){res=1;break;}
		}if(res>=0)l=mid+1;
		else r=mid;
	}ans=l;
	l=1,r=len;
	while(l<r){
		int mid=(l+r)>>1;
		int res=0;
		for(int i=L;i<=R;i++){
			if(sa[mid]+i-L>len){res=1;break;}
			if(s[i]<s[sa[mid]+i-L]){res=-1;break;}
			if(s[i]>s[sa[mid]+i-L]){res=1;break;}
		}if(res>0)l=mid+1;
		else r=mid;
	}return ans-l;
}
int main(){
	scanf("%d",&n);
	for(int i=1;i<=n;i++){
		lef[i]=len+1;
		char c=getchar();
		while(!isalpha(c))c=getchar();
		while(isalpha(c))s[++len]=c,c=getchar();		
		rig[i]=len;s[++len]='{';
	}sa_init(len);
	for(int i=1;i<=n;i++)
		printf("%d\n",Qsum(lef[i],rig[i]));	
	return 0;
}


你可能感兴趣的:(bzoj)