题目链接:POJ - 1743 (不可重叠最长子串)
题意:有N(1<=N<=20000)个音符的序列来表示一首乐曲,每个音符都是1..88范围内的整数,现在要找一个重复的子串,它需要满足如下条件:
1.长度至少为5个音符。
2.在乐曲中重复出现(就是出现过至少两次)。(可能经过转调,“转调”的意思是主题序列中每个音符都被加上或减去了同一个整数值)
3.重复出现的同一主题不能有公共部分。
题解:先求出相邻音符的差,因为转调的话肯定差也是不会变的。然后问题就变成了求不可重叠最长重复子串长度,用后缀数组就可以,重点在不可重叠,因为不能重叠,两个子串要隔至少一个位置,不然就相当于后边子串的首音符和前边子串的尾音符重合了,(因为差值是当前和前一个音符的差值),所以只要将限定条件从max-min>=k改为max-min>k就可以了。二分答案,将问题变为判断是否存在两个长度为k的子串是相同的,且不重叠。如果存在一个区间内的height值都大于等于k,并且存在两个后缀的距离大于k,则说明存在这样的字符串。
倍增法版本:(憨憨的我以为DC3 t了结果是卡输入输出)
1 #include<cstdio> 2 #include<algorithm> 3 #include<queue> 4 #include<iostream> 5 #include<cmath> 6 #include<cstring> 7 using namespace std; 8 9 const int INF=0x3f3f3f3f; 10 const int maxn = 1e6+10; 11 int wa[maxn],wb[maxn],wsf[maxn],wv[maxn],sa[maxn]; 12 int rnk[maxn],height[maxn],s[maxn]; 13 int r[maxn],n; 14 15 //sa:字典序中排第i位的起始位置在str中第sa[i] sa[1~n]为有效值 16 17 //rnk:就是str第i个位置的后缀是在字典序排第几 rnk[0~n-1]为有效值 18 19 //height:字典序排i和i-1的后缀的最长公共前缀 height[2~n]为有效值,第二个到最后一个 20 21 int cmp(int *r,int a,int b,int k) 22 { 23 return r[a]==r[b]&&r[a+k]==r[b+k]; 24 } 25 26 void getsa(int *r,int *sa,int n,int m)//n为添加0后的总长 27 { 28 int i,j,p,*x=wa,*y=wb,*t; 29 for(i=0; i<m; i++) wsf[i]=0; 30 for(i=0; i<=n; i++) wsf[x[i]=r[i]]++; 31 for(i=1; i<m; i++) wsf[i]+=wsf[i-1]; 32 for(i=n; i>=0; i--) sa[--wsf[x[i]]]=i; 33 p=1; 34 j=1; 35 for(; p<=n; j*=2,m=p){ 36 for(p=0,i=n+1-j; i<=n; i++) y[p++]=i; 37 for(i=0; i<=n; i++) if(sa[i]>=j) y[p++]=sa[i]-j; 38 for(i=0; i<=n; i++) wv[i]=x[y[i]]; 39 for(i=0; i<m; i++) wsf[i]=0; 40 for(i=0; i<=n; i++) wsf[wv[i]]++; 41 for(i=1; i<m; i++) wsf[i]+=wsf[i-1]; 42 for(i=n; i>=0; i--) sa[--wsf[wv[i]]]=y[i]; 43 swap(x,y); 44 x[sa[0]]=0; 45 for(p=1,i=1; i<=n; i++) 46 x[sa[i]]=cmp(y,sa[i-1],sa[i],j)? p-1:p++; 47 } 48 } 49 50 void getheight(int *r,int n)//n为添加0后的总长 51 { 52 int i,j,k=0; 53 for(i=1; i<=n; i++) rnk[sa[i]]=i; 54 for(i=0; i<n; i++){ 55 if(k) 56 k--; 57 else 58 k=0; 59 j=sa[rnk[i]-1]; 60 while(r[i+k]==r[j+k]) 61 k++; 62 height[rnk[i]]=k; 63 } 64 } 65 66 bool check(int k) 67 { 68 bool flag=0; 69 int mx=-INF,mn=INF; 70 for(int i=2;i<=n;i++){ 71 if(height[i]>=k){ 72 mn=min(mn,min(sa[i],sa[i-1])); 73 mx=max(mx,max(sa[i],sa[i-1])); 74 if(mx-mn>k) return true; 75 } 76 else mx=-INF,mn=INF; 77 } 78 return false; 79 } 80 81 int main() 82 { 83 ios::sync_with_stdio(false); 84 cin.tie(0); 85 cout.tie(0); 86 while(cin>>n&&n){ 87 for(int i=0;i<n;i++) cin>>s[i]; 88 --n; 89 for(int i=0;i<n;i++) r[i]=s[i+1]-s[i]+100; 90 r[n]=0; 91 getsa(r,sa,n,200); 92 getheight(r,n); 93 int l=0,r=n/2; 94 int ans=0; 95 while(l<=r){ 96 int mid=l+r>>1; 97 if(check(mid)){ 98 ans=mid; 99 l=mid+1; 100 } 101 else r=mid-1; 102 } 103 if(ans>=4) cout<<ans+1<<endl; //因为当前ans是记录的差的长度,串的长度要+1 104 else cout<<0<<endl; 105 } 106 return 0; 107 }
DC3版本:
1 #include<cstdio> 2 #include<algorithm> 3 #include<queue> 4 #include<iostream> 5 #include<cmath> 6 #include<cstring> 7 using namespace std; 8 9 #define F(x) ((x)/3+((x)%3==1?0:tb)) 10 #define G(x) ((x)<tb?(x)*3+1:((x)-tb)*3+2) 11 12 //sa:字典序中排第i位的起始位置在s中第sa[i] 13 14 //rnk:就是s第i个位置的后缀是在字典序排第几 15 16 //height:字典序排i和i-1的后缀的最长公共前缀 17 18 const int INF=0x3f3f3f3f; 19 const int MAXN = 3e6+10; 20 int sa[MAXN]; 21 int rankk[MAXN]; 22 int height[MAXN]; 23 int n; 24 int s[MAXN]; 25 int r[MAXN]; 26 int wa[MAXN],wb[MAXN],wv[MAXN]; 27 int wws[MAXN]; 28 29 void sort(int *r,int *a,int *b,int n,int m) 30 { 31 int i; 32 for(i=0;i<n;i++) wv[i]=r[a[i]]; 33 for(i=0;i<m;i++) wws[i]=0; 34 for(i=0;i<n;i++) wws[wv[i]]++; 35 for(i=1;i<m;i++) wws[i]+=wws[i-1]; 36 for(i=n-1;i>=0;i--) b[--wws[wv[i]]]=a[i]; 37 return; 38 } 39 40 int c0(int *r,int a,int b) 41 { 42 return r[a]==r[b]&&r[a+1]==r[b+1]&&r[a+2]==r[b+2]; 43 } 44 45 int c12(int k,int *r,int a,int b) 46 { 47 if(k==2) return r[a]<r[b]||r[a]==r[b]&&c12(1,r,a+1,b+1); 48 else return r[a]<r[b]||r[a]==r[b]&&wv[a+1]<wv[b+1]; 49 } 50 51 void dc3(int *r,int *sa,int n,int m) 52 { 53 int i,j,*rn=r+n,*san=sa+n,ta=0,tb=(n+1)/3,tbc=0,p; 54 r[n]=r[n+1]=0; 55 for(i=0;i<n;i++) if(i%3!=0) wa[tbc++]=i; 56 sort(r+2,wa,wb,tbc,m); 57 sort(r+1,wb,wa,tbc,m); 58 sort(r,wa,wb,tbc,m); 59 for(p=1,rn[F(wb[0])]=0,i=1;i<tbc;i++) 60 rn[F(wb[i])]=c0(r,wb[i-1],wb[i])?p-1:p++; 61 if(p<tbc) dc3(rn,san,tbc,p); 62 else for(i=0;i<tbc;i++) san[rn[i]]=i; 63 for(i=0;i<tbc;i++) if(san[i]<tb) wb[ta++]=san[i]*3; 64 if(n%3==1) wb[ta++]=n-1; 65 sort(r,wb,wa,ta,m); 66 for(i=0;i<tbc;i++) wv[wb[i]=G(san[i])]=i; 67 for(i=0,j=0,p=0;i<ta && j<tbc;p++) 68 sa[p]=c12(wb[j]%3,r,wa[i],wb[j])?wa[i++]:wb[j++]; 69 for(;i<ta;p++) sa[p]=wa[i++]; 70 for(;j<tbc;p++) sa[p]=wb[j++]; 71 return; 72 } 73 74 void calheight(int *r, int *sa, int n) 75 { 76 int i, j, k = 0; 77 for (i = 1; i <= n; ++i) rankk[sa[i]] = i; 78 for (i = 0; i < n; height[rankk[i++]] = k) 79 for (k ? k-- : 0, j = sa[rankk[i] - 1]; r[i + k] == r[j + k]; ++k); 80 return; 81 } 82 83 bool check(int k) 84 { 85 bool flag=0; 86 int mx=-INF,mn=INF; 87 for(int i=2;i<=n;i++){ 88 if(height[i]>=k){ 89 mn=min(mn,min(sa[i],sa[i-1])); 90 mx=max(mx,max(sa[i],sa[i-1])); 91 if(mx-mn>k) return true; 92 } 93 else mx=-INF,mn=INF; 94 } 95 return false; 96 } 97 98 int main() 99 { 100 ios::sync_with_stdio(false); 101 cin.tie(0); 102 cout.tie(0); 103 while(cin>>n&&n){ 104 for(int i=0;i<n;i++) cin>>s[i]; 105 --n; 106 for(int i=0;i<n;i++) r[i]=s[i+1]-s[i]+100; 107 r[n]=0; 108 dc3(r,sa,n+1,200); 109 calheight(r,sa,n); 110 int l=0,r=n/2; 111 int ans=0; 112 while(l<=r){ 113 int mid=l+r>>1; 114 if(check(mid)){ 115 ans=mid; 116 l=mid+1; 117 } 118 else r=mid-1; 119 } 120 if(ans>=4) cout<<ans+1<<endl; ////因为当前ans是记录的差的长度,串的长度要+1 121 else cout<<0<<endl; 122 } 123 return 0; 124 }