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

POJ 3450 Corporate Identity(KMP)

时间:2016-11-16 14:03:46      阅读:187      评论:0      收藏:0      [点我收藏+]

标签:gets   字符串   main   ict   rpo   names   ++   rate   匹配   

 

【题目链接】 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)

标签:gets   字符串   main   ict   rpo   names   ++   rate   匹配   

原文地址:http://www.cnblogs.com/forever97/p/poj3450.html

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