zoukankan      html  css  js  c++  java
  • POJ3693 Maximum repetition substring [后缀数组 ST表]

    Maximum repetition substring
    Time Limit: 1000MS   Memory Limit: 65536K
    Total Submissions: 9458   Accepted: 2915

    Description

    The repetition number of a string is defined as the maximum number R such that the string can be partitioned into R same consecutive substrings. For example, the repetition number of "ababab" is 3 and "ababa" is 1.

    Given a string containing lowercase letters, you are to find a substring of it with maximum repetition number.

    Input

    The input consists of multiple test cases. Each test case contains exactly one line, which
    gives a non-empty string consisting of lowercase letters. The length of the string will not be greater than 100,000.

    The last test case is followed by a line containing a '#'.

    Output

    For each test case, print a line containing the test case number( beginning with 1) followed by the substring of maximum repetition number. If there are multiple substrings of maximum repetition number, print the lexicographically smallest one.

    Sample Input

    ccabababc
    daabbccaa
    #

    Sample Output

    Case 1: ababab
    Case 2: aa

    Source


    [2016-12-29 00:26]
    从20点写(照代码抄)到现在....
    不想写思路了,看论文吧,那么
    整理一下调试过程的问题:
    STL:
    reverse(s+1,s+1+n) 第一个别写成s
    ST表中:
    1.预处理Log的话是N不是n..........silly
    2.一定要写好了,i和j分清了别顺手写错,找好久还不知道问题出在哪
    SA:
    1.写了个调试函数test.....
    2.通常不需要memset什么,注意重设m的范围吧
     
    4个多小时不太值啊,不过也算是对后缀数组有了点感觉了吧,明天再写两道这样的题
    现在什么错误都不要紧,省选时不要犯就好
     
    [2016-12-29 12:36:21]
    补一下做法:

    重复次数最多的连续重复子串 

    论文:

    先穷举长度 L,然后求长度为 L 的子串最多能连续出现几次。首先连续出现1 次是肯定可以的,所以这里只考虑至少 2 次的情况。假设在原字符串中连续出 现 2 次,记这个子字符串为 S,那么 S 肯定包括了字符 r[0], r[L], r[L*2], r[L*3], ......中的某相邻的两个。所以只须看字符 r[L*i]和 r[L*(i+1)]往前和 往后各能匹配到多远,记这个总长度为 K,那么这里连续出现了 K/L+1 次。最后 看最大值是多少。 

    穷举长度 L 的时间是 n,每次计算的时间是 n/L。所以整个做法的时间复杂度是 O(n/1+n/2+n/3+......+n/n)=O(nlogn)。 

    我们不知道它的长度,所以只能枚举长度

    对于长度L,他的连续重复子串有的话一定覆盖掉s[1+L*i]中的相邻两个,我们把这样的位置成为“关键点”

    所以对于每相邻两个位置(关键点)s[1+L*i]和s[1+L*(i+1)],求出他们往左和往右能匹配多远l和r(我的r是包括了这个关键点),然后在这一段内连续重复的次数step=(l+r)/L+1  

    (因为...自己想想吧,比如对于左端点,移动之后还是相同,还是两个左端点隔了L的距离)

    一个重要的问题是字典序最小

    对于靠左关键点i,向左能延伸l的话,[i-l,i-l+(l+r)%L]这个区间内开始大小重复次数不变((l+r)%L就是说这一个长度是空余的可以随便左右放),所以这一段求rnk最小值作为开始,同样对rnk处理ST表 

     
    [2017-02-06 ]
    复习后缀数组又自己写了一遍.....
    新问题:
    1.求LCP别忘x++,因为是height
    2.大于小于别写错了
    3.反转之后别忘了转回来
    #include <iostream>
    #include <cstdio>
    #include <cstring>
    #include <algorithm>
    using namespace std;
    const int N=1e5+5,INF=1e9;
    int n,m,c[N],t1[N],t2[N];
    char s[N];
    
    int mn[N][17],Log[N],Pow[20];
    void iniST(){
        Pow[0]=1;for(int i=1;i<17;i++) Pow[i]=Pow[i-1]<<1;
        Log[0]=-1;for(int i=1;i<N;i++) Log[i]=Log[i>>1]+1;
    }
    void getST(int mn[N][17],int a[N]){
        for(int i=1;i<=n;i++) mn[i][0]=a[i];
        for(int j=1;j<=Log[n];j++)
            for(int i=1;i+Pow[j]-1<=n;i++)
                mn[i][j]=min(mn[i][j-1],mn[i+Pow[j-1]][j-1]);
    }
    int RMQ(int x,int y){
        if(x>y) swap(x,y);
        int _=Log[y-x+1];
        return min(mn[x][_],mn[y-Pow[_]+1][_]);
    }
    struct SA{
        int sa[N],rnk[N],height[N];
        inline bool cmp(int *r,int a,int b,int j){
            return a+j<=n&&b+j<=n&&r[a]==r[b]&&r[a+j]==r[b+j];
        }
        void getSA(char s[]){
            m=260;
            int *r=t1,*k=t2;
            for(int i=0;i<=m;i++) c[i]=0;
            for(int i=1;i<=n;i++) c[r[i]=s[i]]++;
            for(int i=1;i<=m;i++) c[i]+=c[i-1];
            for(int i=n;i>=1;i--) sa[c[r[i]]--]=i;
    
            for(int j=1;j<=n;j<<=1){
                int p=0;
                for(int i=n-j+1;i<=n;i++) k[++p]=i;
                for(int i=1;i<=n;i++) if(sa[i]>j) k[++p]=sa[i]-j;
    
                for(int i=0;i<=m;i++) c[i]=0;
                for(int i=1;i<=n;i++) c[r[k[i]]]++;
                for(int i=1;i<=m;i++) c[i]+=c[i-1];
                for(int i=n;i>=1;i--) sa[c[r[k[i]]]--]=k[i];
    
                swap(r,k);p=0;r[sa[1]]=++p;
                for(int i=2;i<=n;i++) r[sa[i]]=cmp(k,sa[i],sa[i-1],j)?p:++p;
                if(p>=n) break;m=p;
            }
        }
        void getHeight(char s[]){
            for(int i=1;i<=n;i++) rnk[sa[i]]=i;
            int k=0;
            for(int i=1;i<=n;i++){
                if(k) k--;
                if(rnk[i]==1) continue;
                int j=sa[rnk[i]-1];
                while(i+k<=n&&j+k<=n&&s[i+k]==s[j+k]) k++;
                height[rnk[i]]=k;
            }
        }
        int mn[N][17];
        void ini(char s[]){getSA(s);getHeight(s);getST(mn,height);}
        int lcp(int x,int y){
            x=rnk[x];y=rnk[y];
            if(x>y) swap(x,y);x++;//!!!
            int _=Log[y-x+1];
            return min(mn[x][_],mn[y-Pow[_]+1][_]);
        }
    }a,b;
    
    int cas=0;
    void solve(){
        int lexi=INF,mx=1,al=0,ar=0;
        for(int L=1;L<=n;L++){ 
            for(int i=1;i+L<=n;i+=L){
                int l=b.lcp(n-i+2,n-(i+L)+2),r=a.lcp(i,i+L);
                int step=(l+r)/L+1;
                if(step>mx){
                    mx=step;
                    int _=RMQ(i-l,i-l+(l+r)%L);
                    lexi=_;
                    al=a.sa[_],ar=al+L*step-1;
                }else if(step==mx){
                    int _=RMQ(i-l,i-l+(l+r)%L);
                    if(_<lexi) lexi=_,al=a.sa[_],ar=al+L*step-1;
                }
            }
        }
    
        printf("Case %d: ",++cas);
        reverse(s+1,s+1+n);
        for(int i=al;i<=ar;i++) putchar(s[i]);
        puts("");
    }
    int main(){
        freopen("in","r",stdin);
        iniST();
        while(scanf("%s",s+1)!=EOF){
            if(s[1]=='#') break;
             n=strlen(s+1);
            a.ini(s);
            //for(int i=1;i<=n;i++) printf("a %d %d %d
    ",i,a.rnk[i],a.height[i]);
            reverse(s+1,s+1+n);
            b.ini(s);
            getST(mn,a.rnk);
            solve();
        }
    }
    2.26.2017
     
    #include <iostream>
    #include <cstdio>
    #include <cstring>
    #include <algorithm>
    using namespace std;
    const int N=1e5+5,INF=1e9;
    int n,m,c[N],t1[N],t2[N];
    char s[N];
    inline bool cmp(int *r,int a,int b,int j){
        return a+j<=n&&b+j<=n&&r[a]==r[b]&&r[a+j]==r[b+j];
    }
    int Log[N],Pow[20],mn[N][17];
    
    void iniST(){
        Pow[0]=1;for(int i=1;i<20;i++)Pow[i]=Pow[i-1]<<1;
        Log[0]=-1;for(int i=1;i<=100000;i++)Log[i]=Log[i/2]+1;
    }
    void getST(int mn[N][17],int a[]){
        for(int i=1;i<=n;i++) mn[i][0]=a[i];
        for(int j=1;j<=Log[n];j++)
            for(int i=1;i+Pow[j]-1<=n;i++)
                mn[i][j]=min(mn[i][j-1],mn[i+Pow[j-1]][j-1]);
    }
    
    inline int rmq(int x,int y){
        int t=Log[y-x+1];
        return min(mn[x][t],mn[y-Pow[t]+1][t]);
    }
    
    struct SA{
        int sa[N],rnk[N],height[N],mn[N][17];
        
        void getHeight(){
            int k=0;
            for(int i=1;i<=n;i++) rnk[sa[i]]=i;
            for(int i=1;i<=n;i++){
                if(k) k--;
                if(rnk[i]==1) continue;
                int j=sa[rnk[i]-1];
                while(i+k<=n&&j+k<=n&&s[i+k]==s[j+k]) k++;
                height[rnk[i]]=k;
            }
        }
        void getSA(){
            int *r=t1,*k=t2;
            for(int i=0;i<=m;i++) c[i]=0;
            for(int i=1;i<=n;i++) c[r[i]=s[i]]++;
            for(int i=1;i<=m;i++) c[i]+=c[i-1];
            for(int i=n;i>=1;i--) sa[c[r[i]]--]=i;
            
            for(int j=1;j<=n;j<<=1){
                int p=0;
                for(int i=n-j+1;i<=n;i++) k[++p]=i;
                for(int i=1;i<=n;i++) if(sa[i]>j) k[++p]=sa[i]-j;
                
                for(int i=0;i<=m;i++) c[i]=0;
                for(int i=1;i<=n;i++) c[r[k[i]]]++;
                for(int i=1;i<=m;i++) c[i]+=c[i-1];
                for(int i=n;i>=1;i--) sa[c[r[k[i]]]--]=k[i];
                
                swap(r,k);p=0;r[sa[1]]=++p;
                for(int i=2;i<=n;i++) r[sa[i]]=cmp(k,sa[i],sa[i-1],j)?p:++p;
                if(p>=n) break;m=p;
            }
        }
        
        int lcp(int x,int y){
            x=rnk[x];y=rnk[y];
            if(x>y) swap(x,y);x++;
            int t=Log[y-x+1];
            return min(mn[x][t],mn[y-Pow[t]+1][t]);
        }
        
        void ini(){
            m=300;
            getSA();getHeight();getST(mn,height);
        }
        void test(){
            for(int i=1;i<=n;i++) printf("%c ",s[i]);puts("");
            for(int i=1;i<=n;i++) printf("%d ",rnk[i]);puts("");
            for(int i=1;i<=n;i++) printf("%d ",sa[i]);puts("");
            for(int i=1;i<=n;i++) printf("%d ",height[i]);puts("");
            puts("");
        }
    }a,b;
    int mx,ans,ansl,ansr;
    void solve(int L){//printf("sol %d
    ",L);
        for(int i=1;i+L<=n;i+=L)
            if(s[i]==s[i+L]){
                int r=a.lcp(i,i+L),l=b.lcp(n-i+2,n-i-L+2);
                int step=(l+r)/L+1;//printf("hi %d %d lr %d %d step %d
    ",i,L,l,r,step);
                if(step>mx) mx=step,ans=INF;//,printf("mx %d
    ",mx);
                if(step==mx){
                    int t=rmq(i-l,i-l+(l+r)%L);//printf("t %d
    ",t);
                    if(t<ans){
                        ans=t;
                        ansl=a.sa[t],ansr=ansl+mx*L-1;
                    }
                }
            }
    }
    
    int main(){
        int cas=0;
        iniST();
        while(scanf("%s",s+1)!=EOF){
            if(s[1]=='#') break;
            n=strlen(s+1);
            a.ini();//a.test();
            reverse(s+1,s+1+n);
            b.ini();//b.test();
            
            getST(mn,a.rnk);
            reverse(s+1,s+1+n);
            mx=1;ans=INF;ansl=ansr=0;
            for(int i=1;i<=n;i++)
                if(a.rnk[i]<ans) ans=a.rnk[i],ansl=ansr=i;
            for(int L=1;L<=n;L++) solve(L);
            
            printf("Case %d: ",++cas);
            for(int i=ansl;i<=ansr;i++) putchar(s[i]);
            puts("");
            
        }
    }
     
     
  • 相关阅读:
    Codevs 1038 一元三次方程求解 NOIP 2001(导数 牛顿迭代)
    Bzoj 3942: [Usaco2015 Feb]Censoring(kmp)
    Bzoj 1355: [Baltic2009]Radio Transmission(kmp)
    Bzoj 2242: [SDOI2011]计算器(BSGS)
    Cogs 1345. [ZJOI2013] K大数查询(树套树)
    Cogs 58. 延绵的山峰(st表)
    洛谷 P2251 质量检测(st表)
    洛谷 P3382 【模板】三分法(三分 二分)
    Hihocoder #1142 : 三分·三分求极值
    P1967 货车运输
  • 原文地址:https://www.cnblogs.com/candy99/p/6231412.html
Copyright © 2011-2022 走看看