题目描述:

给你两个字符串,求它们最长公共子串的长度,如果不存在公共子串则输出0。

样例输入:

yeshowmuchiloveyoumydearmotherreallyicannotbelieveit
yeaphowmuchiloveyoumydearmother

样例输出:

27

题解:

SAM果题。

代码+SAM理解记录附下:

//
// Title : SAMachine
// Date : 10.03.2016
// Test : CODEVS-3160
// Complexity : O(n)
//
/*
对于子串匹配/处理等问题——
解决办法:后缀自动机
*/
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath> #ifdef WIN32
#define LL "%I64d"
#else
#define LL "%lld"
#endif #ifdef CT
#define debug(...) printf(__VA_ARGS__)
#else
#define debug(...)
#endif #define R register
#define getc() (S==T&&(T=(S=B)+fread(B,1,1<<15,stdin),S==T)?EOF:*S++)
#define gmax(_a, _b) ((_a) > (_b) ? (_a) : (_b))
#define gmin(_a, _b) ((_a) < (_b) ? (_a) : (_b))
#define cmax(_a, _b) (_a < (_b) ? _a = (_b) : 0)
#define cmin(_a, _b) (_a > (_b) ? _a = (_b) : 0)
char B[1<<15],*S=B,*T=B;
inline int FastIn()
{
R char ch;R int cnt=0;R bool minus=0;
while (ch=getc(),(ch < '0' || ch > '9') && ch != '-') ;
ch == '-' ?minus=1:cnt=ch-'0';
while (ch=getc(),ch >= '0' && ch <= '9') cnt = cnt * 10 + ch - '0';
return minus?-cnt:cnt;
}
#define maxn 250010
char str[maxn] ;
int ans ;
struct SAM
{
int p , np , q , nq;
int cnt , last;
int a[maxn][26] , l[maxn] , fa[maxn];
SAM()
{
last=++cnt;
}
void extend(R int c)
{
p = last ; // p表示添加前的最后一个状态
np = last = ++cnt; // np表示当前的状态
l[np] = l[p] + 1; // 当前的长度为上一个状态的长度+1
for (; !a[p][c] && p ; ) a[p][c] = np , p = fa[p] ; // 如果p的fail指针的trie边没有被占用,那么就将其trie边连到np上
if (!p) fa[np] = 1 ; // 如果当前这单个字符没有出现过,那么就把当前状态的fail指针设为1(init,初始状态)
else // 如果p的fail指针被占用了,那么就————
{
q = a[p][c]; // q表示被占用掉的那个状态
if (l[p] + 1 == l[q] ) fa[np] = q; // 如果这个状态刚好是当前状态的上一个字符的状态,那么就意味着出现了两个相同的字符,此时就直接将fail指针建到q上
else // 否则就
{
nq = ++cnt ; // 新建一个节点
l[nq] = l[p] + 1 ; // 新建的这个节点将会的长度等于当前状态的长度
memcpy (a[nq] , a[q] , sizeof (a[q]) ); // 将当前占用掉的那个状态的trie边连到新的节点上
fa[nq] = fa[q] ; // 新建节点的fail指针指向上一个状态
fa[np] = fa[q] = nq ; // 当前的节点和被占用的节点的fail指针指向新的节点
while (a[p][c] == q) a[p][c] = nq , p = fa[p] ; // 把所有trie边指向q的点的trie边连向nq
}
}
}
void build()
{
scanf( "%s\n" , str );
for (R int i = 0 ; str[i] ; i++ ) extend ( str[i] - 'a' ) ;
}
void solve()
{
scanf( "%s\n" , str );
R int tmp = 0;
for (R int i = 0 ; str[i] ; i++ )
{
R int c = str[i] - 'a';
if ( a[p][c] ) p = a[p][c] , tmp ++ ;
else
{
for ( ; p && !a[p][c] ; ) p = fa[p] ;
if (!p) p = 1 ,tmp = 0;
else tmp = l[p] + 1 , p = a[p][c];
}
cmax( ans , tmp );
}
printf("%d\n",ans );
}
}sam;
int main()
{
sam.build();
sam.solve();
return 0;
}

【spoj2774】最长公共子串的更多相关文章

  1. [Data Structure] LCSs——最长公共子序列和最长公共子串

    1. 什么是 LCSs? 什么是 LCSs? 好多博友看到这几个字母可能比较困惑,因为这是我自己对两个常见问题的统称,它们分别为最长公共子序列问题(Longest-Common-Subsequence ...

  2. HDU 1503 带回朔路径的最长公共子串

    http://acm.hdu.edu.cn/showproblem.php?pid=1503 这道题又WA了好几次 在裸最长公共子串基础上加了回溯功能,就是给三种状态各做一个 不同的标记.dp[n][ ...

  3. 最长公共子序列PK最长公共子串

    1.先科普下最长公共子序列 & 最长公共子串的区别: 找两个字符串的最长公共子串,这个子串要求在原字符串中是连续的.而最长公共子序列则并不要求连续. (1)递归方法求最长公共子序列的长度 1) ...

  4. 动态规划(一)——最长公共子序列和最长公共子串

    注: 最长公共子序列采用动态规划解决,由于子问题重叠,故采用数组缓存结果,保存最佳取值方向.输出结果时,则自顶向下建立二叉树,自底向上输出,则这过程中没有分叉路,结果唯一. 最长公共子串采用参考串方式 ...

  5. 字符串hash + 二分答案 - 求最长公共子串 --- poj 2774

    Long Long Message Problem's Link:http://poj.org/problem?id=2774 Mean: 求两个字符串的最长公共子串的长度. analyse: 前面在 ...

  6. 后缀数组(模板题) - 求最长公共子串 - poj 2774 Long Long Message

    Language: Default Long Long Message Time Limit: 4000MS   Memory Limit: 131072K Total Submissions: 21 ...

  7. 最长公共子串 NYOJ 36

    http://acm.nyist.net/JudgeOnline/problem.php?pid=36 最长公共子序列 时间限制:3000 ms  |  内存限制:65535 KB 难度:3   描述 ...

  8. POJ 2217 (后缀数组+最长公共子串)

    题目链接: http://poj.org/problem?id=2217 题目大意: 求两个串的最长公共子串,注意子串是连续的,而子序列可以不连续. 解题思路: 后缀数组解法是这类问题的模板解法. 对 ...

  9. poj1159 dp最长公共子串

    //Accepted 204 KB 891 ms //dp最长公共子串 //dp[i][j]=max(dp[i-1][j],dp[i][j-1]) //dp[i][j]=max(dp[i][j],dp ...

随机推荐

  1. Xcode增加头文件搜索路径的方法

    Xcode增加头文件搜索路径的方法 以C++工程为例: 在Build Settings 页面中的Search Paths一节就是用来设置头文件路径. 相关的配置项用红框框起来了,共有三个配置项: He ...

  2. [转帖]查看ubuntu 各系统的内核版本

    查看ubuntu 各系统的内核版本 https://www.cnblogs.com/ranxf/p/6923311.html /etc/issue /proc/version 1.查看ubuntu版本 ...

  3. CSS3与页面布局学习总结——多种页面布局

    一.负边距与浮动布局 1.1.负边距 所谓负边距就是margin取负值的情况,如margin:-40px:margin-left:-100%.当一个元素与另一个元素margin取负值时将拉近距离.常见 ...

  4. Maven - Maven3实战学习笔记(3)使用maven构建Web应用

    1.jetty-maven-plugin自动化测试Web应用工具 <plugin> <groupId>org.mortbay.jetty</groupId> < ...

  5. PHPStorm运行PHP代码

    1.完成php代码后,点击空白处,右上角会出现 chrome.firefox等浏览器的图标,选择一个电脑上已有图标 2.默认情况下浏览器会提示“bad gateway”,此时需要配置phpstorm对 ...

  6. python 9*9乘法口诀 猜数字游戏

  7. ubuntu中apache的ssl证书配置及url重写

    一.https原理 借用网上的图(图片来源: https://www.cnblogs.com/xiohao/p/9054355.html ),用到了对称加密和非对称加密.    二.ubuntu的ap ...

  8. 错误代码errno值的含义

    错误代码errno值的含义 查看错误代码errno是调试程序的一个重要方法.当C api函数发生异常时,一般会将errno变量(需include errno.h)赋一个整数值,不同的值表示不同的含义, ...

  9. SpringBoot集合Linux的FastDFS与Nginx上传图片测试错误com.github.tobato.fastdfs.exception.FdfsConnectException: 无法获取服务端连接资源:can't create connection to/192.168.1.104:22122

    报错 com.github.tobato.fastdfs.exception.FdfsConnectException: 无法获取服务端连接资源:can't create connection to/ ...

  10. 状态码是canceled

    timeout : 1000 给ajax配置如上属性 $.ajax({ type:"post", url:"pro/savePro", timeout : 10 ...