最长重复不重叠字串

把排序后的后缀分成若干组,其中每组的后缀之间的height 值都

不小于k。例如,字符串为“aabaaaab”,当k=2 时,后缀分成了4 组,如图5所示。


。然
后对于每组后缀,只须判断每个后缀的sa 值的最大值和最小值之差是否不小于
k。如果有一组满足,则说明存在,否则不存在。整个做法的时间复杂度为
O(nlogn)。

[cpp]  view plain copy
  1. //最长重复不重叠子串  
  2. #include <iostream>  
  3. #include <string>  
  4. #include <cmath>  
  5. #include <map>  
  6. using namespace std;  
  7. const int N=30000;  
  8. int wa[N],wb[N],wv[N],wsum[N];  
  9. int height[N],sa[N],rank[N];  
  10. int n;  
  11. int str[N];  
  12. int cmp(int *r,int a,int b,int l)  
  13. {  
  14.     return r[a]==r[b] && r[a+l]==r[b+l];  
  15. }  
  16. void da(int *r,int *sa,int n,int m)  
  17. {  
  18.     int i,j,p,*x=wa,*y=wb,*t;  
  19.     for(i=0;i<m;i++)  
  20.         wsum[i]=0;  
  21.     for(i=0;i<n;i++)  
  22.         wsum[x[i]=r[i]]++;  
  23.     for(i=1;i<m;i++)  
  24.         wsum[i]+=wsum[i-1];  
  25.     for(i=n-1;i>=0;i--)  
  26.         sa[--wsum[x[i]]]=i;  
  27.     for(j=1,p=1;p<n;j*=2,m=p)  
  28.     {  
  29.         for(p=0,i=n-j;i<n;i++)  
  30.             y[p++]=i;  
  31.         for(i=0;i<n;i++)  
  32.             if(sa[i]>=j)  
  33.                 y[p++]=sa[i]-j;  
  34.         for(i=0;i<n;i++)  
  35.             wv[i]=x[y[i]];  
  36.         for(i=0;i<m;i++)  
  37.             wsum[i]=0;  
  38.         for(i=0;i<n;i++)  
  39.             wsum[wv[i]]++;  
  40.         for(i=1;i<m;i++)  
  41.             wsum[i]+=wsum[i-1];  
  42.         for(i=n-1;i>=0;i--)  
  43.             sa[--wsum[wv[i]]]=y[i];  
  44.         for(t=x,x=y,y=t,p=1,x[sa[0]]=0,i=1;i<n;i++)  
  45.             x[sa[i]]=cmp(y,sa[i-1],sa[i],j)?p-1:p++;  
  46.     }  
  47. }  
  48. void calheight(int *r,int *sa,int n)  
  49. {  
  50.     int i,j,k=0;  
  51.     for(i=0;i<=n;i++)  
  52.         rank[sa[i]]=i;  
  53.     for(i=0;i<n;height[rank[i++]]=k)  
  54.         for(k?k--:0,j=sa[rank[i]-1];r[i+k]==r[j+k];k++);  
  55. }  
  56. bool check(int x)  
  57. {  
  58.     int i;  
  59.     int mmin=N,mmax=0;  
  60.     for(i=2;i<=n;i++)  
  61.     {  
  62.         if(height[i]>=x)  
  63.         {  
  64.             mmin=sa[i]<mmin?sa[i]:mmin;  
  65.             mmax=sa[i]>mmax?sa[i]:mmax;  
  66.             if(mmax-mmin>=x)  
  67.                 return 1;  
  68.         }  
  69.         else  
  70.         {  
  71.             mmin=mmax=sa[i];  
  72.         }  
  73.     }  
  74.     return 0;  
  75. }  
  76. int main()  
  77. {  
  78.     int i,j,k,l,r,mid;  
  79.     int ans;  
  80.     while(~scanf("%d",&n) && n)  
  81.     {  
  82.         for(i=0;i<n;i++)  
  83.             scanf("%d",&str[i]);  
  84.         for(i=n-1;i>0;i--)  
  85.             str[i]=str[i]-str[i-1]+90;  
  86.         str[n]=0;  
  87.         str[0]=199;  
  88.         da(str,sa,n+1,200);  
  89.         calheight(str,sa,n);  
  90.         ans=0;  
  91.         l=1;  
  92.         r=n/2;  
  93.         ans=0;  
  94.         while(l<=r)  
  95.         {  
  96.             mid=(l+r)>>1;  
  97.             if(check(mid))  
  98.             {  
  99.                 ans=mid;  
  100.                 l=mid+1;  
  101.             }  
  102.             else  
  103.             {  
  104.                 r=mid-1;  
  105.             }  
  106.         }  
  107.         ans++;  
  108.         if(ans<5)  
  109.             ans=0;  
  110.         printf("%d\n",ans);  
  111.     }  
  112.     return 0;  
  113. }  
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值