又学到一个\(SAM\)的新套路QvQ

思路

考虑用其中的一个串建个\(SAM\),然后用其他的串在上面匹配,匹配时更新答案

首先有一个全局变量\(len\),表示当前已匹配的长度。假设目前在点\(u\),转移方式如下(根节点为\(1\)):

如果没有对应的转移边,就走后缀连接,\(u=suflink(u)\),并令\(len=maxlen(suflink(u))\)。否则走对应的转移边,同时\(len++\)。如果一直没有对应的转移边,即到最后发现\(u=0\),就把\(u\)置为\(1\),\(len\)置为\(0\),并开始下个字符的匹配

开一个数组\(mx\)记录每个结点被匹配时的\(len\)最大是多少,全部匹配完后还要拓扑排序一遍,把每个结点的\(mx\)上传给其\(parent\ tree\)上的祖先。对于一个结点\(u\),它所代表的\(lcs\)长度为每个字符串匹配完后\(mx\)中的最小值,每次更新一下就行了

代码

  1. #include <bits/stdc++.h>
  2. using namespace std;
  3. #define N 100000
  4. int m, n, root = 1, nid = 1, last = 1, maxlen[2*N+5], ch[2*N+5][26], link[2*N+5], mx[2*N+5], mn[2*N+5], len;
  5. int tmp[2*N+5], a[2*N+5];
  6. void insert(int c) {
  7. int cur = ++nid;
  8. maxlen[cur] = maxlen[last]+1;
  9. while(last && !ch[last][c]) ch[last][c] = cur, last = link[last];
  10. if(!last) link[cur] = root;
  11. else {
  12. int p = last, q = ch[last][c];
  13. if(maxlen[q] == maxlen[p]+1) link[cur] = q;
  14. else {
  15. int clone = ++nid;
  16. maxlen[clone] = maxlen[p]+1;
  17. for(int i = 0; i < 26; ++i) ch[clone][i] = ch[q][i];
  18. link[clone] = link[q]; link[q] = link[cur] = clone;
  19. while(p && ch[p][c] == q) ch[p][c] = clone, p = link[p];
  20. }
  21. }
  22. last = cur;
  23. }
  24. void radixSort() {
  25. memset(tmp, 0, sizeof tmp);
  26. for(int i = 1; i <= nid; ++i) tmp[maxlen[i]]++;
  27. for(int i = 1; i <= m; ++i) tmp[i] += tmp[i-1];
  28. for(int i = 1; i <= nid; ++i) a[tmp[maxlen[i]]--] = i;
  29. for(int i = nid; i >= 1; --i)
  30. mx[link[a[i]]] = max(mx[link[a[i]]], min(maxlen[link[a[i]]], mx[a[i]])), mn[a[i]] = min(mn[a[i]], mx[a[i]]);
  31. }
  32. void calc(char *s) {
  33. n = strlen(s);
  34. memset(mx, 0, sizeof mx);
  35. int u = root;
  36. len = 0;
  37. for(int i = 0; i < n; ++i) {
  38. while(u && !ch[u][s[i]-'a']) u = link[u], len = maxlen[u];
  39. if(!u) u = root;
  40. else {
  41. u = ch[u][s[i]-'a'];
  42. len++;
  43. mx[u] = max(mx[u], len);
  44. }
  45. }
  46. radixSort();
  47. }
  48. int main() {
  49. char s[N+5];
  50. scanf("%s", s);
  51. m = strlen(s);
  52. for(int i = 0; i < m; ++i) insert(s[i]-'a');
  53. memset(mn, 0x3f, sizeof mn);
  54. while(~scanf("%s", s)) calc(s);
  55. int ans = 0;
  56. for(int i = 1; i <= nid; ++i) ans = max(ans, mn[i]);
  57. printf("%d\n", ans);
  58. return 0;
  59. }

例题

SP1811

SP1812

SP10570

[SDOI2008]Sandy的卡片

SAM求多个串的最长公共子串的更多相关文章

  1. SPOJ 1811 Longest Common Substring(求两个串的最长公共子串 || 或者n个串)

    http://www.spoj.com/problems/LCS/ 题目:求两个串的最长公共子串 参考:https://www.cnblogs.com/autoint/p/10345276.html: ...

  2. SPOJ 1811 Longest Common Substring (后缀自动机第一题,求两个串的最长公共子串)

    题目大意: 给出两个长度小于等于25W的字符串,求它们的最长公共子串. 题目链接:http://www.spoj.com/problems/LCS/ 算法讨论: 二分+哈希, 后缀数组, 后缀自动机. ...

  3. 多个串的最长公共子串 SPOJ - LCS2 后缀自动机

    题意: 求多个串的最长公共子串 这里用的是O(n)的后缀自动机写法 我后缀数组的专题有nlog(n)写法的 题解: 对于其中的一个串建立后缀自动机 然后对于后缀自动机上面的每一个节点求出每一个节点最长 ...

  4. 【poj1226-出现或反转后出现在每个串的最长公共子串】后缀数组

    题意:求n个串的最长公共子串,子串出现在一个串中可以是它的反转串出现.总长<=10^4. 题解: 对于每个串,把反转串也连进去.二分长度,分组,判断每个组. #include<cstdio ...

  5. SPOJ LCS2 多个串的最长公共子串

    这里串最多有10个,找所有串的最长公共子串 这里后缀自动机做,以第一个串建立后缀自动机,后面的串一个个去匹配,每次得到当前串在可到达状态上所能得到的最长后缀长度 拿所有串匹配后得到的结果进行计算 #i ...

  6. 求两个字符串的最长公共子串——Java实现

    要求:求两个字符串的最长公共子串,如“abcdefg”和“adefgwgeweg”的最长公共子串为“defg”(子串必须是连续的) public class Main03{ // 求解两个字符号的最长 ...

  7. [URAL-1517][求两个字符串的最长公共子串]

    Freedom of Choice URAL - 1517 Background Before Albanian people could bear with the freedom of speec ...

  8. POJ 2774 求两个串的最长公共前缀 | 后缀数组

    #include<cstdio> #include<algorithm> #include<cstring> #define N 200005 using name ...

  9. [SPOJ1812]Longest Common Substring II 后缀自动机 多个串的最长公共子串

    题目链接:http://www.spoj.com/problems/LCS2/ 其实两个串的LCS会了,多个串的LCS也就差不多了. 我们先用一个串建立后缀自动机,然后其它的串在上面跑.跑的时候算出每 ...

随机推荐

  1. HotSpot 虚拟机垃圾回收算法实现

    作为使用范围最广的虚拟机之一HotSpot,必须对垃圾回收算法的执行效率有严格的考量,只有这样才能保证虚拟机高效运行 枚举根节点 从可达性分析中从 GC Roots 节点找引用链这个操作为例,可以作为 ...

  2. Web后端 JAVA学习之路

    1.Java分类 Java按应用来分,可以分为J2ME(手机版),J2SE(标准版),J2EE(企业版)三部分. ・J2ME:已经被安卓开发取代. ・J2SE:Java的核心类,其中包括桌面应用,但一 ...

  3. vue路由懒加载 及import

  4. 图像的膨胀与腐蚀——OpenCV与C++的具体实现

    目录 1. 膨胀与腐蚀的原理 2. 膨胀的具体实现 1) OpenCV实现 2) C/C++实现 3) 验证与结果 3. 腐蚀的具体实现 1. 膨胀与腐蚀的原理 膨胀与腐蚀是数学形态学在图像处理中最基 ...

  5. Android为TV端助力之WebView开发踩坑一

    在Android清单配置文件里面 自定义application时,在4.4系统上面不能加上一个属性,见下图 否则界面将不会显示任何数据,在更高或者更低的系统上面没有测试!

  6. 深入理解group by 语句的执行顺序 from→where→group by→select(含聚合函数)

    由于之前没有对group by 语句的执行顺序(执行原理)做深入的了解,所以导致在实际应用过程中出现了一些问题.举个简单的粟子,比如一个表testA中的所有数据如下图: 我现在想从testA中查询us ...

  7. Python使用Plotly绘图工具,绘制甘特图

    今天来讲一下如何使用Python 的绘图工具Plotly来绘制甘特图的方法 甘特图大家应该了解熟悉,就是通过条形来显示项目的进度.时间安排等相关情况的. 我们今天来学习一下,如何使用ployly来绘制 ...

  8. 【工作分解法】IT人,你的工作“轻松”么?

    一.前言 假如读者是一个老板,下面有两位员工,工作难度一样,完成量一样,人品和责任心也一样.一位每天加班加点,废寝忘食的工作:而另外一位每天在座位上喝着咖啡,非常的轻松自如的工作.您会更器重哪一位? ...

  9. Audio播放

    <audio controls="controls" id="warnAudio" hidden> <source src="~/m ...

  10. MySql 触发器的新增、修改、删除的创建

    MySql 触发器与SQL server 触发器不同: SQL Server   使用 inserted.deleted 代表被触发的数据. MySQL NEW代表触发后的新数据行,Old代表当前触发 ...