*题目描述:
给出几个由小写字母构成的单词,求它们最长的公共子串的长度。
任务:
l 读入单词
l 计算最长公共子串的长度
l 输出结果
*输入:
文件的第一行是整数 n,1<=n<=5,表示单词的数量。接下来n行每行一个单词,只由小写字母组成,单词的长度至少为1,最大为2000。
*输出:
仅一行,一个整数,最长公共子串的长度。
*样例输入:
3
abcb
bca
acbc
*样例输出:

*题解:
SAM果题。
做法1:
对第一个串构造后缀自动机,然后剩下的每个串放进去匹配。按照val将节点排序后得到SAM的dfs序,每次记录当前节点的最多匹配的长度,最后每个节点向它的父亲节点统计答案。
做法2:(orz_immortalCO)
将所有的串连起来,两个串中间隔一个字符,然后把这个大串的SAM构造出来,SAM的节点上打上标记。然后把SAM的fail树构出来,这个fail树就是原串反串的后缀树。然后在树上DP就好了。
*代码:
做法1:

#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__)
#define setfile()
#else
#define debug(...)
#define filename ""
#define setfile() freopen(filename".in", "r", stdin); freopen(filename".out", "w", stdout);
#endif #define R register
#define getc() (S == T && (T = (S = B) + fread(B, 1, 1 << 15, stdin), S == T) ? EOF : *S++)
#define dmax(_a, _b) ((_a) > (_b) ? (_a) : (_b))
#define dmin(_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 20010
char str[maxn];
int next[maxn][26], fa[maxn], mx[maxn], tot = 1, cur = 1;
inline void extend(R int c)
{
R int p = cur, np = ++tot;
mx[cur = np] = mx[p] + 1;
for ( ; p && !next[p][c]; p = fa[p])
next[p][c] = np;
if (!p) fa[np] = 1;
else
{
R int q = next[p][c];
if (mx[q] == mx[p] + 1) fa[np] = q;
else
{
R int nq = ++tot;
mx[nq] = mx[p] + 1;
memcpy(next[nq], next[q], sizeof(next[nq]));
fa[nq] = fa[q];
fa[q] = fa[np] = nq;
for ( ; next[p][c] == q; p = fa[p])
next[p][c] = nq;
}
}
}
int ans[maxn], v[maxn], q[maxn], len[maxn];
inline void pre()
{
for (R int i = 1; i <= tot; ++i) ans[i] = mx[i];
for (R int i = 1; i <= tot; ++i) v[mx[i]]++;
for (R int i = 1; i <= tot; ++i) v[i] += v[i - 1];
for (R int i = tot; i; --i) q[v[mx[i]]--] = i;
}
inline void solve()
{
scanf("%s", str + 1);
memset(len, 0, sizeof(len));
R int l = strlen(str + 1), p = 1, tmp = 0;
for (R int i = 1; i <= l; ++i)
{
R int c = str[i] - 'a';
for ( ; p && !next[p][c]; p = fa[p]) ;
if (!p) p = 1, tmp = 0;
else tmp = dmin(tmp, mx[p]) + 1, p = next[p][c];
cmax(len[p], tmp);
}
for (R int i = tot; i; --i) cmax(len[fa[q[i]]], len[q[i]]);
for (R int i = 1; i <= tot; ++i) cmin(ans[i], len[i]);
}
int main()
{
// setfile();
R int n;
scanf("%d", &n);
scanf("%s", str + 1);
R int l = strlen(str + 1);
for (R int i = 1; i <= l; ++i) extend(str[i] - 'a');
pre();
for (R int i = 1; i < n; ++i) solve();
R int fil_ans = 0;
for (R int i = 1; i <= tot; ++i)
cmax(fil_ans, ans[i]);
printf("%d\n", fil_ans);
return 0;
}

【bzoj2946】[Poi2000]公共串的更多相关文章

  1. [bzoj2946][Poi2000]公共串_后缀数组_二分

    公共串 bzoj-2946 Poi-2000 题目大意:给定$n$个字符串,求他们的最长公共子串. 注释:$1\le n\le 5$,$1\le minlen<maxlen\le 2000$. ...

  2. [BZOJ2946] [Poi2000]公共串解题报告|后缀数组

    给出几个由小写字母构成的单词,求它们最长的公共子串的长度. 单词个数<=5,每个单词长度<=2000     尽管最近在学的是SAM...但是看到这个题还是忍不住想写SA... (其实是不 ...

  3. SPOJ1812: LCS2 - Longest Common Substring II & BZOJ2946: [Poi2000]公共串

    [传送门:SPOJ1811&BZOJ2946] 简要题意: 给出若干个字符串,求出这些字符串的最长公共子串 题解: 后缀自动机 这两道题的区别只是在于一道给出了字符串个数,一个没给,不过也差不 ...

  4. BZOJ2946 [Poi2000]公共串(后缀自动机)

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

  5. BZOJ2946 Poi2000 公共串 【后缀自动机】

    Description 给出几个由小写字母构成的单词,求它们最长的公共子串的长度. 任务: l 读入单词 l 计算最长公共子串的长度 l 输出结果 Input 文件的第一行是整数 n,1<=n& ...

  6. [BZOJ2946][Poi2000]公共串解题报告|后缀自动机

    鉴于SAM要简洁一些...于是又写了一遍这题... 不过很好呢又学到了一些新的东西... 这里是用SA做这道题的方法 首先还是和两个字符串的一样,为第一个字符串建SAM 然后每一个字符串再在这个SAM ...

  7. 【二分答案】【分块答案】【字符串哈希】【set】bzoj2946 [Poi2000]公共串

    我们二分/分块枚举答案x,暴力把除了最短的字符串以外的其他字符串的x长度子串哈希搞出来,分别扔到set里. 然后暴力枚举最短的字符串的x长度字串,查看是否在全部的set里出现过. #include&l ...

  8. bzoj2946: [Poi2000]公共串

    SAM处女题qwq #include <iostream> #include <cstdio> #include <cstring> #include <cm ...

  9. bzoj2946 [Poi2000]公共串(SA,SAM)

    [题意] 多串求LCS.   [思路]   主要是想找一下SAM的优越感 :) velui good 后缀数组划分height需要注意不少细节 <_<,然后不停debug   [代码]   ...

  10. 【二分答案】【哈希表】【字符串哈希】bzoj2946 [Poi2000]公共串

    二分答案,然后搞出hash值扔到哈希表里.期望复杂度O(n*log(n)). <法一>next数组版哈希表 #include<cstdio> #include<cstri ...

随机推荐

  1. 匿名函数及paramiko模块

    1.匿名函数 随着程序代码的不断增加,起名字其实也是非常困难的一件事 一些简单的功能完全没必要用def函数,匿名函数足矣 def test(x,y): return x+y res = test(1, ...

  2. Linux 系统安装 python

    Centos 7 Centos 7 安装 python3 (不要卸载python2 因为yum 要用) https://phoenixnap.com/kb/how-to-install-python- ...

  3. 【Qt开发】Linux下Qt开发环境的安装与集成

    近期工作需要在Linux下用Qt进行C++开发,所以就在linux下尝试装QT开发环境.本人用的linux是CentOS 6.5.现在对安装过程做出总结.有两种安装方式,下面分别详述: 1 图形化安装 ...

  4. 【Qt开发】【VS开发】VS2010+Qt开发环境搭建

    QT与JAVA有点类似,也是一种跨平台的软件(当然在windows平台和linux平台需要安装相应的QT开发环境和运行库,类似于JAVA在不同平台下的虚拟机JVM环境),因此对于某些需要同时支持win ...

  5. Nginx/Nginx配置文件

    nginx.conf配置文件 mac目录位置:/usr/local/etc/nginx/ ubuntu目录位置:/etc/nginx nginx.conf配置 /* 全局块:配置影响nginx全局的指 ...

  6. 基于weui loading插件封装

    <!-- Loading.vue --> <template> <div id="loadingToast" v-if="show" ...

  7. fragment嵌套子fragment

    场景: 需求是在expandablelistview的item长按会弹出一个对话框,对话框顶部一个选项卡,下面是fragment作相应的切换.对话框是用DialogFragment实现的. 问题: j ...

  8. 第四周课程总结与第二次实验报告(Java简单类与对象)

    1.写一个名为Rectangle的类表示矩形.其属性包括宽width.高height和颜色color,width和height都是double型的,而color则是String类型的.要求该类具有: ...

  9. Python入门之format()方法

    在此列出format()方法的一些基本使用: >>> '{}{}{}'.format('圆周率是',3.1415926,'...') '圆周率是3.1415926...' >& ...

  10. 批处理遍历文件夹执行git pull

    echo off & color 0A for /d %%f in (D:\www\*) do ( D: cd %%f chdir git pull ) pause 遍历D:\www\这个文件 ...