码迷,mamicode.com
首页 > 其他好文 > 详细

【POJ1743】不可重叠最长重复子串

时间:2015-12-15 18:08:44      阅读:207      评论:0      收藏:0      [点我收藏+]

标签:

 

  题意:求一个字符串里两个不重叠的最长重复子串

 

 技术分享

 

技术分享

 

 

 代码如下:

技术分享
  1 #include<cstdio>
  2 #include<cstdlib>
  3 #include<cstring>
  4 #include<iostream>
  5 using namespace std;
  6 
  7 int sa[20010],rank[20010],y[20010],Rsort[20010];
  8 int wr[20010],a[20010],height[20010],n;
  9 
 10 bool cmp(int k1,int k2,int ln){return wr[k1]==wr[k2] && wr[k1+ln]==wr[k2+ln];}
 11 
 12 void get_sa(int m) 
 13 {
 14   int i,k,p,ln;
 15 
 16   memcpy(rank,a,sizeof(rank));
 17 
 18   memset(Rsort,0,sizeof(Rsort));
 19   for (i=1;i<=n;i++) Rsort[rank[i]]++;
 20   for (i=1;i<=m;i++) Rsort[i]+=Rsort[i-1];
 21   for (i=n;i>=1;i--) sa[Rsort[rank[i]]--]=i; 
 22   
 23   ln=1; p=0; 
 24   while (p<n)
 25   {
 26       for (k=0,i=n-ln+1;i<=n;i++) y[++k]=i;
 27       for (i=1;i<=n;i++) if (sa[i]>ln) y[++k]=sa[i]-ln;
 28       for (i=1;i<=n;i++) wr[i]=rank[y[i]];
 29       
 30       memset(Rsort,0,sizeof(Rsort));
 31       for (i=1;i<=n;i++) Rsort[wr[i]]++;
 32       for (i=1;i<=m;i++) Rsort[i]+=Rsort[i-1];
 33       for (i=n;i>=1;i--) sa[Rsort[wr[i]]--]=y[i];
 34 
 35       memcpy(wr,rank,sizeof(wr));   
 36       p=1; rank[sa[1]]=1;
 37       for (i=2;i<=n;i++)
 38       {
 39          if (!cmp(sa[i],sa[i-1],ln)) p++;
 40          rank[sa[i]]=p;
 41       }
 42       m=p; ln*=2;
 43   }
 44   a[0]=sa[0]=0;
 45 }
 46 
 47 void get_he()
 48 {
 49   int i,j,k=0;
 50   for (i=1;i<=n;i++)
 51   {
 52      j=sa[rank[i]-1];
 53      if (k) k--;
 54     
 55      while (a[j+k]==a[i+k]) k++;
 56      height[rank[i]]=k;
 57   }
 58 }
 59 
 60 bool check(int k)
 61 {
 62     int i,maxx=0,minn=20010;
 63     for(i=1;i<=n;i++)
 64     {
 65         if(height[i]<k) maxx=minn=sa[i];
 66         else
 67         {
 68             if(sa[i]>maxx) maxx=sa[i];
 69             if(sa[i]<minn) minn=sa[i];
 70             if(maxx-minn>k) return 1;
 71         }
 72     }
 73     return 0;
 74 }
 75 
 76 int hd_work()
 77 {
 78     int l,r,mid,ans=0;
 79     l=1;r=n;
 80     while(l<=r)
 81     {
 82         mid=(l+r)/2;
 83         if(check(mid)) 
 84         {
 85           l=mid+1;
 86           ans=mid;
 87         }
 88         else r=mid-1;
 89     }
 90     if(ans>=4) ans++;
 91     else ans=0;
 92     return ans;
 93 }
 94 
 95 int main()
 96 {
 97     int i,a1,a2;
 98     while(1)
 99     {
100         scanf("%d",&n);
101         if(n==0) break;
102         scanf("%d",&a1);
103         for(i=2;i<=n;i++)
104         {
105             scanf("%d",&a2);
106             a[i-1]=a2-a1+88;
107             a1=a2;
108         } 
109         n--;
110         get_sa(20010);
111         get_he();
112         printf("%d\n",hd_work());
113     }
114 }
poj1743

 

2015-12-15 17:09:09

【POJ1743】不可重叠最长重复子串

标签:

原文地址:http://www.cnblogs.com/Konjakmoyu/p/5048791.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!