BZOJ 2946: [Poi2000]公共串( 后缀自动机 )

时间:2022-08-27 14:58:32

BZOJ 2946: [Poi2000]公共串( 后缀自动机 )

一个串建后缀自动机, 其他串在上面跑, 然后用当前串跑的去更新全部

---------------------------------------------------------------------------

#include<cstdio>
#include<cstring>
#include<algorithm>
 
using namespace std;
 
const int maxn = 10009;
const int n = 26;
 
char s[maxn];
 
struct Node {
Node *fa, *ch[n];
int len, mx, ans;
} pool[maxn], *pt, *Root, *Last;
 
Node* newNode(int l) {
pt->fa = NULL;
pt->ans = pt->len = l;
memset(pt->ch, 0, sizeof pt->ch);
return pt++;
}
 
void InitSAM() {
pt = pool;
Last = Root = newNode(0);
}
 
void Extend(int c) {
Node *p = Last, *np = newNode(p->len + 1);
for(; p && !p->ch[c]; p = p->fa)
p->ch[c] = np;
if(!p)
np->fa = Root;
else {
Node *q = p->ch[c];
if(q->len == p->len + 1)
np->fa = q;
else {
Node *nq = newNode(p->len + 1);
memcpy(nq->ch, q->ch, sizeof q->ch);
nq->fa = q->fa;
q->fa = np->fa = nq;
for(; p && p->ch[c] == q; p = p->fa)
p->ch[c] = nq;
}
}
Last = np;
}
 
inline void upd(int &x, int t) {
if(t > x)
x = t;
}
 
int main() {
InitSAM();
int T; scanf("%d", &T);
scanf("%s", s);
for(int i = 0, n = strlen(s); i < n; i++) Extend(s[i] - 'a');
if(T == 1) {
printf("%d\n", strlen(s));
return 0;
}
while(--T) {
for(Node *o = pt; o-- != pool; ) o->mx = 0;
scanf("%s", s);
Node *t = Root;
for(int i = 0, n = strlen(s), lth = 0; i < n; i++) {
int c = s[i] - 'a';
if(t->ch[c])
upd((t = t->ch[c])->mx,++lth);
else {
for(; t && !t->ch[c]; t = t->fa);
if(!t) {
lth = 0; t = Root;
} else {
lth = t->len + 1;
upd((t = t->ch[c])->mx, lth);
}
}
}
for(Node *o = pt; o-- != pool; )
if(o->fa) upd(o->fa->mx, o->mx);
for(Node *o = pt; o-- != pool; )
o->ans = min(o->ans, o->mx);
}
int ans = 0;
for(Node *o = pt; o-- != pool; upd(ans, o->ans));
printf("%d\n", ans);
return 0;
}

---------------------------------------------------------------------------

2946: [Poi2000]公共串

Time Limit: 3 Sec  Memory Limit: 128 MB
Submit: 327  Solved: 143
[Submit][Status][Discuss]

Description

 
       给出几个由小写字母构成的单词,求它们最长的公共子串的长度。
任务:
l        读入单词
l        计算最长公共子串的长度
l        输出结果
 

Input

 
文件的第一行是整数 n,1<=n<=5,表示单词的数量。接下来n行每行一个单词,只由小写字母组成,单词的长度至少为1,最大为2000。
 

Output

仅一行,一个整数,最长公共子串的长度。
 

Sample Input

3
abcb
bca
acbc

Sample Output

HINT

Source

BZOJ 2946: [Poi2000]公共串( 后缀自动机 )的更多相关文章

  1. bzoj 2946 &lbrack;Poi2000&rsqb;公共串——后缀自动机

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2946 对每个串都建一个后缀自动机,然后 dfs 其中一个自动机,记录同步的话在别的自动机上走 ...

  2. BZOJ 2946 &lbrack;Poi2000&rsqb;公共串 ——后缀自动机

    任意选择一个串作为模式串,构建出后缀自动机. 然后用其他的串在后缀自动机上跑匹配. 然后就到了理解后缀自动机性质的时候. 在某一个节点的最大值是可以沿着parent树上传的. 然后用dp[i][j]表 ...

  3. BZOJ 2946 POI2000 公共串 后缀自动机(多串最长公共子串)

    题意概述:给出N个字符串,每个串的长度<=2000(雾...可能是当年的年代太久远机子太差了),问这N个字符串的最长公共子串长度为多少.(N<=5) 抛开数据结构,先想想朴素做法. 设计一 ...

  4. BZOJ 2946&colon; &lbrack;Poi2000&rsqb;公共串

    2946: [Poi2000]公共串 Time Limit: 3 Sec  Memory Limit: 128 MBSubmit: 787  Solved: 342[Submit][Status][D ...

  5. 【bzoj2946】&lbrack;Poi2000&rsqb;公共串 后缀自动机

    [Poi2000]公共串 Time Limit: 3 Sec  Memory Limit: 128 MBSubmit: 1386  Solved: 620[Submit][Status][Discus ...

  6. BZOJ 2946 &lbrack;Poi2000&rsqb;公共串 &lpar;二分&plus;Hash&sol;二分&plus;后缀数组&sol;后缀自动机&rpar;

    求多串的最长公共字串. 法1: 二分长度+hash 传送门 法2: 二分+后缀数组 传送门 法3: 后缀自动机 拿第一个串建自动机,然后用其他串在上面匹配.每次求出SAM上每个节点的最长匹配长度后,再 ...

  7. BZOJ2946 &lbrack;Poi2000&rsqb;公共串&lpar;后缀自动机&rpar;

    Description          给出几个由小写字母构成的单词,求它们最长的公共子串的长度. 任务: l        读入单词 l        计算最长公共子串的长度 l        输 ...

  8. bzoj 2946&colon; &lbrack;Poi2000&rsqb;公共串【SAM】

    对第一个串建SAM,把剩下的串在上面跑,每次跑一个串的时候在SAM的端点上记录匹配到这的最大长度,然后对这些串跑的结果取min,然后从这些节点的min中取max就是答案 注意在一个点更新后它的祖先也会 ...

  9. 【BZOJ 2946】 2946&colon; &lbrack;Poi2000&rsqb;公共串 (SAM)

    2946: [Poi2000]公共串 Time Limit: 3 Sec  Memory Limit: 128 MBSubmit: 1063  Solved: 469 Description      ...

随机推荐

  1. 2DToolkit官方文档中文版打地鼠教程(三):Sprite Collections 精灵集合

    这是2DToolkit官方文档中 Whack a Mole 打地鼠教程的译文,为了减少文中过多重复操作的翻译,以及一些无必要的句子,这里我假设你有Unity的基础知识(例如了解如何新建Sprite等) ...

  2. linux 批量删除进程

    2016年11月18日 13:11:10 星期五 ps -ef | grep pname | awk '{print $2}' | xargs kill 解释: 杀掉所有包含 'pname' 的进程

  3. WinForm中TreeView控件实现鼠标拖动节点&lpar;可实现同级节点位置互换,或拖到目标子节点&rpar;

    ;//1:不同级, 不为1:拖同级 private void treeView1_ItemDrag(object sender, ItemDragEventArgs e) { if (e.Button ...

  4. 判断百度某一经纬度的地图颜色值python

    from PIL import Image import MySQLdb import os import urllib import time from multiprocessing.dummy ...

  5. 【BZOJ】1856&colon; &lbrack;Scoi2010&rsqb;字符串

    http://www.lydsy.com/JudgeOnline/problem.php?id=1856 题意:把n个1和m个0组成字符串,要求在组成的字符串中,任意的前k个字符1的个数不能少于0的个 ...

  6. mapreduce 实现矩阵乘法

    import java.io.IOException; import org.apache.hadoop.conf.Configuration; import org.apache.hadoop.fs ...

  7. mybatis中几种typeHandler的定义使用

    1.存储到数据库, 将LONG数组转换成字符串;从数据库获取数据, 将字符串转为LONG数组 package com.winturn.utils.handler; import java.sql.Ca ...

  8. TensorFlow学习笔记(UTF-8 问题解决 UnicodeDecodeError&colon; &&num;39&semi;utf-8&&num;39&semi; codec can&&num;39&semi;t decode byte 0xff in position 0&colon; invalid start byte)

    我使用VS2013  Python3.5  TensorFlow 1.3  的开发环境 UnicodeDecodeError: 'utf-8' codec can't decode byte 0xff ...

  9. python爬虫学习记录——各种软件&sol;库的安装

    Ubuntu18.04安装python3-pip 1.apt-get update更新源 2,ubuntu18.04默认安装了python3,但是pip没有安装,安装命令:apt install py ...

  10. 关于Swift中的指针的那些事

    前言 在Objective-c的世界中,一切对象都是指针.它是一种运行时语言,具体指针的对象类型将会在运行时,由系统分配.这样虽然*,但是却并不安全. Swift世界就不一样了,Swift的世界很安 ...