后缀自动机(SAM) :SPOJ LCS - Longest Common Substring

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了后缀自动机(SAM) :SPOJ LCS - Longest Common Substring相关的知识,希望对你有一定的参考价值。

LCS - Longest Common Substring

no tags 

 

A string is finite sequence of characters over a non-empty finite set Σ.

In this problem, Σ is the set of lowercase letters.

Substring, also called factor, is a consecutive sequence of characters occurrences at least once in a string.

Now your task is simple, for two given strings, find the length of the longest common substring of them.

Here common substring means a substring of two or more strings.

Input

The input contains exactly two lines, each line consists of no more than 250000 lowercase letters, representing a string.

Output

The length of the longest common substring. If such string doesn‘t exist, print "0" instead.

Example

Input:
alsdfkjfjkdsal
fdjskalajfkdsla

Output:
3

  这题就是求最长公共子串,于是我直接套用了SAM。
 1 #include <iostream>
 2 #include <cstdio>
 3 #include <cstring>
 4 using namespace std;
 5 const int MAXN=1000010;
 6 struct SAM{
 7     int fa[MAXN],ch[MAXN][27],len[MAXN],cnt,last;
 8     void Init()
 9     {
10         last=cnt=1;
11     }
12     void Insert(int c)
13     {
14         int p=last,np=last=++cnt;
15         len[np]=len[p]+1;
16         while(!ch[p][c]&&p)
17             ch[p][c]=np,p=fa[p];
18         if(!p)fa[np]=1;
19         else{
20             int q=ch[p][c];
21             if(len[q]==len[p]+1)
22                 fa[np]=q;
23             else{
24                 int nq=++cnt;len[nq]=len[p]+1;
25                 memcpy(ch[nq],ch[q],sizeof(ch[p]));
26                 fa[nq]=fa[q];fa[q]=fa[np]=nq;
27                 while(ch[p][c]==q)
28                     ch[p][c]=nq,p=fa[p];
29             }    
30         }    
31     }
32 }sam; 
33 char str1[MAXN],str2[MAXN];
34 int main()
35 {
36     int len1=0,len2=0,ans=0;
37     scanf("%s%s",str1,str2);
38     sam.Init();
39     while(str1[len1])
40         sam.Insert(str1[len1++]-`);
41     int node=1,tmp=0;
42     for(;str2[len2];len2++){
43         int c=str2[len2]-`;
44         if(sam.ch[node][c])
45             node=sam.ch[node][c],tmp++;
46         else{
47             while(node&&!sam.ch[node][c])
48                 node=sam.fa[node];
49             if(node==0)
50                 node=1,tmp=0;
51             else
52                 tmp=sam.len[node]+1,node=sam.ch[node][c];        
53         }
54         ans=max(ans,tmp);
55     }
56     printf("%d\n",ans);
57     return 0;
58 }

 

以上是关于后缀自动机(SAM) :SPOJ LCS - Longest Common Substring的主要内容,如果未能解决你的问题,请参考以下文章

SPOJ 1812 LCS2 [后缀自动机]

SPOJ 1812 LCS2 - Longest Common Substring II (后缀自动机)两种做法

spoj1811 & spoj1812 - LCS1 & LCS2sam

SPOJ - LCS 后缀自动机入门

SPOJ LCS2 - Longest Common Substring II 后缀自动机 多个串的LCS

SPOJ 1812 LCS2 后缀自动机