【SP1811】 LCS - Longest Common Substring(后缀自动机)
2024-08-21 04:54:43
题目链接
对第一个串建出\(SAM\),然后用第二个串去匹配。
如果能往下走就往下走,不能的话就跳parent tree的父亲,直到能走为止。如果跳到\(0\)了还是不能走,重新匹配。
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
const int MAXN = 1000010;
struct SAM{
int ch[26];
int len, fa;
}sam[MAXN << 1];
int las = 1, cnt = 1;
inline void add(int c){
int p = las; int np = las = ++cnt;
sam[np].len = sam[p].len + 1;
for(; p && !sam[p].ch[c]; p = sam[p].fa) sam[p].ch[c] = np;
if(!p) sam[np].fa = 1;
else{
int q = sam[p].ch[c];
if(sam[q].len == sam[p].len + 1) sam[np].fa = q;
else{
int nq = ++cnt; sam[nq] = sam[q];
sam[nq].len = sam[p].len + 1;
sam[q].fa = sam[np].fa = nq;
for(; p && sam[p].ch[c] == q; p = sam[p].fa) sam[p].ch[c] = nq;
}
}
}
char a[MAXN], b[MAXN];
int ans;
int main(){
scanf("%s", a + 1);
scanf("%s", b + 1);
int lena = strlen(a + 1), lenb = strlen(b + 1);
for(int i = 1; i <= lena; ++i)
add(a[i] - 'a');
int p = 1, len = 0;
for(int i = 1; i <= lenb; ++i){
if(sam[p].ch[b[i] - 'a'])
++len, p = sam[p].ch[b[i] - 'a'];
else{
while(!sam[p].ch[b[i] - 'a'] && p) p = sam[p].fa;
if(!p) p = 1, len = 0;
else len = sam[p].len + 1, p = sam[p].ch[b[i] - 'a'];
}
ans = max(ans, len);
}
printf("%d\n", ans);
return 0;
}
最新文章
- 在页面的el表达式是如何判断null的
- linux split (分割文件)命令
- HTML前端--各种小案例
- bzoj1015 星球大战
- VS Extension: WPF : 使用全局方式 设置 窗体 foreground background
- block 解析 - block变量
- 菜鸟进阶Android Touch事件传递(四)
- 扩展js,实现c#中的string.format方便拼接字符串
- IT项目角色标准定义
- JAVA中默认的编码方式
- Nero8刻录引导系统光盘镜像图文教程
- node 学习(一)
- 搭建日志收集系统时使用客户端连接etcd遇到的问题
- A、B两个线程交替打印1 -- 100
- Arcmap内容列表刷新
- 搜索路径---PYTHONPATH 变量
- C# socket 发送图片和文件
- DJI Mobile SDK 新教程
- 用自定义的RoundImageView来实现圆形图片(可加边框)
- 【BZOJ4337】BJOI2015 树的同构 括号序列