题面

问题描述

给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,„,Sn)都是T的子串。

输入

第一行是一个正整数n(n<=12),表示给定的字符串的个数。以下的n行,每行有一个全由大写字母组成的字符串。每个字符串的长度不超过50.

输出

只有一行,为找到的最短的字符串T。在保证最短的前提下,如果有多个字符串都满足要求,那么必须输出按字典序排列的第一个。

Sample Input

  1. 2
  2. ABCD
  3. BCDABC

Sample Output

  1. ABCDABC

题解

AC自动机第一题...

实际上这题用到的是trie图, 状压DP即可. 每个节点记录在什么状态下被经过, 用于去重. 时间复杂度和空间复杂度都是\(2^nnL\)

  1. #include <cstdio>
  2. #include <cstring>
  3. #include <deque>
  4. const int N = 12, LEN = 50;
  5. int n;
  6. char ans[N * LEN];
  7. struct ACautomaton
  8. {
  9. struct node
  10. {
  11. node *suc[26], *fl;
  12. int vst[1 << N], ed;
  13. inline node()
  14. {
  15. for(int i = 0; i < 26; ++ i)
  16. suc[i] = NULL;
  17. memset(vst, 0, sizeof(vst));
  18. ed = -1;
  19. }
  20. }*rt;
  21. inline ACautomaton()
  22. {
  23. rt = new node;
  24. rt->fl = rt;
  25. }
  26. inline void insert(char *str, int len, int id)
  27. {
  28. node *u = rt;
  29. for(int i = 0; i < len; u = u->suc[str[i] - 'A'], ++ i)
  30. if(u->suc[str[i] - 'A'] == NULL)
  31. u->suc[str[i] - 'A'] = new node;
  32. u->ed = id;
  33. }
  34. inline void build()
  35. {
  36. static std::deque<node*> que;
  37. que.clear();
  38. for(int i = 0; i < 26; ++ i)
  39. if(rt->suc[i] != NULL)
  40. rt->suc[i]->fl = rt, que.push_back(rt->suc[i]);
  41. for(; ! que.empty(); que.pop_front())
  42. {
  43. node *u = que.front();
  44. for(int i = 0; i < 26; ++ i)
  45. if(u->suc[i] != NULL)
  46. {
  47. node *p = u->fl;
  48. for(; p != rt && p->suc[i] == NULL; p = p->fl);
  49. u->suc[i]->fl = p->suc[i] == NULL ? p : p->suc[i];
  50. que.push_back(u->suc[i]);
  51. }
  52. for(int i = 0; i < 26; ++ i)
  53. if(u->suc[i] == NULL)
  54. u->suc[i] = u->fl->suc[i];
  55. }
  56. }
  57. struct state
  58. {
  59. node *u;
  60. int lst, rec, c;
  61. inline state(node *_u, int _lst, int _rec, int _c)
  62. {
  63. u = _u, lst = _lst, rec = _rec, c = _c;
  64. }
  65. inline state() {}
  66. };
  67. inline void work()
  68. {
  69. static state que[(1 << N) * N * LEN];
  70. int L = 0, R = 0;
  71. que[R ++] = state(rt, -1, 0, -1);
  72. for(; ; L ++)
  73. {
  74. state cur = que[L];
  75. node *u = cur.u;
  76. int rec = cur.rec;
  77. if(~ u->ed)
  78. rec |= 1 << u->ed, u->vst[rec] = 1;
  79. if(rec == (1 << n) - 1)
  80. break;
  81. for(int i = 0; i < 26; ++ i)
  82. if(u->suc[i] != NULL && ! u->suc[i]->vst[rec])
  83. que[R ++] = state(u->suc[i], L, rec, i), u->suc[i]->vst[rec] = 1;
  84. }
  85. int len = 0;
  86. for(; L; L = que[L].lst)
  87. ans[len ++] = 'A' + que[L].c;
  88. for(int i = len - 1; ~ i; -- i)
  89. putchar(ans[i]);
  90. }
  91. }ACA;
  92. int main()
  93. {
  94. #ifndef ONLINE_JUDGE
  95. freopen("BZOJ1195.in", "r", stdin);
  96. freopen("BZOJ1195.out", "w", stdout);
  97. #endif
  98. scanf("%d\n", &n);
  99. for(int i = 0; i < n; ++ i)
  100. {
  101. static char str[LEN];
  102. scanf("%s", str);
  103. ACA.insert(str, strlen(str), i);
  104. }
  105. ACA.build();
  106. ACA.work();
  107. }

HNOI 2006 BZOJ 1195 最短母串的更多相关文章

  1. [BZOJ 1195] 最短母串

    Link:https://www.lydsy.com/JudgeOnline/problem.php?id=1195 Solution: 看到数据范围n<=12,就要往状压DP上想 为了保证后项 ...

  2. bzoj 1195: [HNOI2006]最短母串 爆搜

    1195: [HNOI2006]最短母串 Time Limit: 10 Sec  Memory Limit: 32 MBSubmit: 894  Solved: 288[Submit][Status] ...

  3. BZOJ 1195: [HNOI2006]最短母串

    1195: [HNOI2006]最短母串 Time Limit: 10 Sec  Memory Limit: 32 MBSubmit: 1346  Solved: 450[Submit][Status ...

  4. bzoj 1195 [HNOI2006]最短母串 bfs 状压 最短路 AC自动机

    LINK:最短母串 求母串的问题.不适合SAM. 可以先简化问题 考虑给出的n个字符串不存在包含关系. 那么 那么存在的情况 只可能有 两个字符串拼接起来能表示另外一个字符串 或者某个字符串的后缀可以 ...

  5. 【状态压缩dp】1195: [HNOI2006]最短母串

    一个清晰的思路就是状压dp:不过也有AC自动机+BFS的做法 Description 给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,„,Sn)都是T ...

  6. 【loj10061】最短母串

    #10061. 「一本通 2.4 练习 4」最短母串 内存限制:512 MiB 时间限制:1000 ms 标准输入输出 题目类型:传统 评测方式:文本比较 上传者: 1bentong 提交    提交 ...

  7. [BZOJ1195]最短母串

    1195: [HNOI2006]最短母串 Time Limit: 10 Sec  Memory Limit: 32 MB Description 给定n个字符串(S1,S2,„,Sn),要求找到一个最 ...

  8. 2782: [HNOI2006]最短母串

    2782: [HNOI2006]最短母串 Time Limit: 1 Sec  Memory Limit: 128 MBSubmit: 3  Solved: 2[Submit][Status][Web ...

  9. BZOJ1195[HNOI2006]最短母串——AC自动机+BFS+状态压缩

    题目描述 给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,„,Sn)都是T的子串. 输入 第一行是一个正整数n(n<=12),表示给定的字符串的 ...

随机推荐

  1. codeforce GYM 100741 A Queries

    A. Queries time limit per test:0.25 s memory limit per test:64 MB input:standard input output:standa ...

  2. MOTCF 没时间解释了 条件竞争漏洞

    moctf 没时间解释了 条件竞争漏洞 题目链接 条件竞争: 在本题目中,上传文件的时候服务器无条件的接收任何类型的文件,但是你上传之后服务器会给你的文件内容修改为too slow. 比如你上传了一句 ...

  3. 【LoadRunner】如何对GIS服务器进行性能测试

    1.需求了解 首先确定对gis服务器压测的测试范围,形成具体的测试用例,gis平台都是通过网页端的javascript api调用的gis集群服务接口,通过LR录制上一步中的业务操作,找到javasc ...

  4. JS 关于 URL 的编码或解码方法

    URL的合法字符 URL的合法字符表示再浏览器的地址栏中不会被转义的字符,有两种: URL元字符:分号(;),逗号(’,’),斜杠(/),问号(?),冒号(:),at(@),&,等号(=),加 ...

  5. Python 字节与字符串的转换

    html = urlopen("http://www.cnblogs.com/ryanzheng/p/9665224.html") bsObj = BeautifulSoup(ht ...

  6. leetcode NO.349 两个数组的交集 (python实现)

    来源 https://leetcode-cn.com/problems/intersection-of-two-arrays/ 题目描述 给定两个数组,写一个函数来计算它们的交集. 例子: 给定 nu ...

  7. idea热部署设置(复制)

    提出问题 IntelliJ IDEA工具如何设置热部署??? 解决问题 我的IDEA的版本是:IntelliJ IDEA 14.0.2 第一步:打开tomcat配置 这里写图片描述 第二步: 这里写图 ...

  8. 函数的apply、call方法和length属性

    函数的apply.call方法和length属性JavaScript为函数对象定义了两个方法:apply和call,它们的作用都是将函数绑定到另外一个对象上去运行,两者仅在定义参数的方式有所区别:Fu ...

  9. quagga源码学习--BGP协议的初始化

    quagga支持BGP-4,BGP-4+协议,支持多协议(mpls,isis,ospf等等)以及单播,组播路由的导入和分发. 具体的协议,这里就不附录了,网络上有很多资料,或者RFC. 协议源码的学习 ...

  10. 【bzoj3834】[Poi2014]Solar Panels 数论

    题目描述 Having decided to invest in renewable energy, Byteasar started a solar panels factory. It appea ...