POJ 3450 Corporate Identity(KMP)

Posted forever97‘s blog

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了POJ 3450 Corporate Identity(KMP)相关的知识,希望对你有一定的参考价值。

 

【题目链接】 http://poj.org/problem?id=3450

 

【题目大意】

  求k个字符串的最长公共子串,如果有多个答案,则输出字典序最小的。

 

【题解】

  我们对第一个串的每一个后缀和其余所有串做kmp,取匹配最小值的最大值就是答案。

 

【代码】

#include <cstring>
#include <cstdio>
#include <algorithm>
const int N=4050,M=210; 
using namespace std;
int nxt[M],n;
char dict[N][M];
void get_nxt(char *a,int n){
    int i,j;
    for(nxt[0]=j=-1,i=1;i<n;nxt[i++]=j){
        while(~j&&a[j+1]!=a[i])j=nxt[j];
        if(a[j+1]==a[i])j++;
    }
}
int LongestPre(char *s,int len){
    get_nxt(s,len);
    for(int i=1;i<n;i++){
        char *p=dict[i];
        int ans=0;
        for(int j=-1;*p;p++){
            while(~j&&s[j+1]!=*p)j=nxt[j];
            if(s[j+1]==*p){
                j++; ans=max(ans,j+1);
            }if(j==len-1)j=nxt[j];
        }len=min(len,ans);
    }return len;
}
int main(){
    while(scanf("%d",&n)&&n){
        getchar();
        for(int i=0;i<n;i++)gets(dict[i]);
        int len=strlen(dict[0]),ans=0,pos=0;
        for(int i=0;i<len;i++){
            int tmp=LongestPre(dict[0]+i,len-i);
            if(tmp>=ans){
                if(tmp>ans)ans=tmp,pos=i;
                else{
                    bool flag=1;
                    for(int t=0;t<ans;t++){
                        if(dict[0][pos+t]>dict[0][i+t])break;
                        else if(dict[0][pos+t]<dict[0][i+t]){flag=0;break;}
                    }if(flag)pos=i;
                }
            }
        }if(ans){
            for(int i=0;i<ans;i++)putchar(dict[0][pos+i]);
                puts("");
        }else puts("IDENTITY LOST");
    }return 0;
}

  

以上是关于POJ 3450 Corporate Identity(KMP)的主要内容,如果未能解决你的问题,请参考以下文章

POJ 3450 Corporate Identity(KMP)

POJ3450 Corporate Identity

POJ3080 POJ3450Corporate Identity(广义后缀自动机||后缀数组||KMP)

POJ 3450--Corporate IdentityKMP &amp;&amp; 枚举

POJ - 3450

POJ3450题解——暴力orKMP