uva11107

题意

输入 n 个 DNA 序列,求出长度最大的字符串,使得它在超过一半的 DNA 序列中连续出现。如果有多解,按字典序输出。

分析

论文

后缀数组经典题。加深几个关键数组的印象。

和 poj2774 一样,都是要去连接字符串,保证分隔符不能和字符串内的字符相同,且不能重复。

为什么要连接呢?因为求后缀数组实际是对后缀字符串进行排序,那么有公共前缀子串的后缀字符串会尽可能的排在一起,不同的分隔符保证公共子串不会扩散到别的串上。而 height 数组对应的就是相邻 sa 数组的 lcp ( 最长公共前缀 )。根据选择的最大长度 m,可以将连续的且 lcp 长度大于等于 m 的后缀子串分到一组,要去掉那些在同一个原串里的子串,用一个标记数组标记当前字符属于哪个原串。最后统计个数是否大于一半即可。

这种求最大、最小应该想到和二分法有关。

code

  1. #include<cstdio>
  2. #include<cstring>
  3. #include<set>
  4. #include<algorithm>
  5. using namespace std;
  6. const int MAXN = 2e5 + 10;
  7. char s[MAXN];
  8. int sa[MAXN], t[MAXN], t2[MAXN], c[MAXN], n; // n 为 字符串长度 + 1,s[n - 1] = 0
  9. int rnk[MAXN], height[MAXN];
  10. // 构造字符串 s 的后缀数组。每个字符值必须为 0 ~ m-1
  11. void build_sa(int m) {
  12. int i, *x = t, *y = t2;
  13. for(i = 0; i < m; i++) c[i] = 0;
  14. for(i = 0; i < n; i++) c[x[i] = s[i]]++;
  15. for(i = 1; i < m; i++) c[i] += c[i - 1];
  16. for(i = n - 1; i >= 0; i--) sa[--c[x[i]]] = i;
  17. for(int k = 1; k <= n; k <<= 1) {
  18. int p = 0;
  19. for(i = n - k; i < n; i++) y[p++] = i;
  20. for(i = 0; i < n; i++) if(sa[i] >= k) y[p++] = sa[i] - k;
  21. for(i = 0; i < m; i++) c[i] = 0;
  22. for(i = 0; i < n; i++) c[x[y[i]]]++;
  23. for(i = 0; i < m; i++) c[i] += c[i - 1];
  24. for(i = n - 1; i >= 0; i--) sa[--c[x[y[i]]]] = y[i];
  25. swap(x, y);
  26. p = 1; x[sa[0]] = 0;
  27. for(i = 1; i < n; i++)
  28. x[sa[i]] = y[sa[i - 1]] == y[sa[i]] && y[sa[i - 1] + k] == y[sa[i] + k] ? p - 1 : p++;
  29. if(p >= n) break;
  30. m = p;
  31. }
  32. }
  33. void getHeight() {
  34. int i, j, k = 0;
  35. for(i = 0; i < n; i++) rnk[sa[i]] = i;
  36. for(i = 0; i < n - 1; i++) {
  37. if(k) k--;
  38. j = sa[rnk[i] - 1];
  39. while(s[i + k] == s[j + k]) {
  40. k++;
  41. }
  42. height[rnk[i]] = k;
  43. }
  44. }
  45. // 保证 s[n-1] = 0 且前面非 0 // 也就是说空串在最前
  46. // sa[0] = n - 1,sa[i] 有效的只有 [1, n-1] ( 因为前面的 n 加 1 了 )表示第 i 位的是谁( 以第几个字符开始的字符串后缀 )
  47. // height[i] 有效的只有 [2, n-1] 表示 lcp(sa[i], sa[i-1]) 最长公共前缀
  48. char s1[MAXN];
  49. int id[MAXN];
  50. int check(int c, int m) {
  51. set<int> S;
  52. S.insert(id[sa[1]]);
  53. for(int i = 2; i < n; i++) {
  54. while(i < n && height[i] >= m) {
  55. S.insert(id[sa[i]]);
  56. i++;
  57. }
  58. if(2 * S.size() > c) return 1;
  59. S.clear();
  60. S.insert(id[sa[i]]);
  61. }
  62. return 0;
  63. }
  64. void print(int c, int m) {
  65. set<int> S;
  66. S.insert(id[sa[1]]);
  67. for(int i = 2; i < n; i++) {
  68. while(i < n && height[i] >= m) {
  69. S.insert(id[sa[i]]);
  70. i++;
  71. }
  72. if(2 * S.size() > c) {
  73. int bgn = sa[i - 1];
  74. for(int j = 0; j < m; j++) {
  75. printf("%c", s[bgn + j]);
  76. }
  77. puts("");
  78. }
  79. S.clear();
  80. S.insert(id[sa[i]]);
  81. }
  82. }
  83. int main() {
  84. int c;
  85. int f = 1;
  86. while(scanf("%d", &c) && c) {
  87. memset(s, 0, sizeof s);
  88. if(!f) puts("");
  89. else f = 0;
  90. int bound = 1;
  91. for(int i = 0; i < c; i++) {
  92. scanf("%s", s1);
  93. int l = strlen(s), l1 = strlen(s1);
  94. for(int j = 0; j < l1; j++) {
  95. s[j + l] = s1[j];
  96. id[j + l] = i;
  97. }
  98. if(bound == 97) bound = 123;
  99. s[l + l1] = bound++; // 分隔符
  100. id[l + l1] = i;
  101. s[l + l1 + 1] = 0;
  102. }
  103. if(c == 1) {
  104. puts(s1); continue;
  105. }
  106. n = strlen(s) + 1; // 保证 s[n-1] = 0
  107. build_sa(128);
  108. getHeight();
  109. int l = 0, r = 1000, mid, ans = 0;
  110. while(l <= r) {
  111. mid = (l + r) / 2;
  112. if(check(c, mid)) { ans = mid; l = mid + 1; }
  113. else r = mid - 1;
  114. }
  115. if(ans == 0) puts("?");
  116. else print(c, ans);
  117. }
  118. return 0;
  119. }

uva11107(后缀数组)的更多相关文章

  1. uva11107 后缀数组

    题意给了n个串 然后计算 这些串中的子串在大于1/2的串中出现 求出这个串的最长长度. 将这些串用一个每出现的不同的字符拼起来 ,然后二分找lcp #include <iostream> ...

  2. UVA11107 Life Forms --- 后缀数组

    UVA11107 Life Forms 题目描述: 求出出现在一半以上的字符串内的最长字符串. 数据范围: \(\sum len(string) <= 10^{5}\) 非常坑的题目. 思路非常 ...

  3. 【UVA11107 训练指南】Life Forms【后缀数组】

    题意 输入n(n<=100)个字符串,每个字符串长度<=1000,你的任务是找出一个最长的字符串使得超过一半的字符串都包含这个字符串. 分析 训练指南上后缀数组的一道例题,据说很经典(估计 ...

  4. 后缀数组的倍增算法(Prefix Doubling)

    后缀数组的倍增算法(Prefix Doubling) 文本内容除特殊注明外,均在知识共享署名-非商业性使用-相同方式共享 3.0协议下提供,附加条款亦可能应用. 最近在自学习BWT算法(Burrows ...

  5. BZOJ 4199: [Noi2015]品酒大会 [后缀数组 带权并查集]

    4199: [Noi2015]品酒大会 UOJ:http://uoj.ac/problem/131 一年一度的“幻影阁夏日品酒大会”隆重开幕了.大会包含品尝和趣味挑战两个环节,分别向优胜者颁发“首席品 ...

  6. BZOJ 1692: [Usaco2007 Dec]队列变换 [后缀数组 贪心]

    1692: [Usaco2007 Dec]队列变换 Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 1383  Solved: 582[Submit][St ...

  7. POJ3693 Maximum repetition substring [后缀数组 ST表]

    Maximum repetition substring Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 9458   Acc ...

  8. POJ1743 Musical Theme [后缀数组]

    Musical Theme Time Limit: 1000MS   Memory Limit: 30000K Total Submissions: 27539   Accepted: 9290 De ...

  9. 后缀数组(suffix array)详解

    写在前面 在字符串处理当中,后缀树和后缀数组都是非常有力的工具. 其中后缀树大家了解得比较多,关于后缀数组则很少见于国内的资料. 其实后缀数组是后缀树的一个非常精巧的替代品,它比后缀树容易编程实现, ...

随机推荐

  1. IIS Express mime type 列表。

    C:\Users\Administrator\Documents\IISExpress\config\applicationhost.config -------------------------- ...

  2. python作业:模拟登陆(第一周)

    模拟登陆作业需求: 1. 用户输入帐号密码进行登陆 2. 用户信息保存在文件内 3. 用户密码输入错误三次后锁定用户 额外实现功能: 1.提示输入错误次数 2.输入已锁定用户会提示 3.用户不存在会提 ...

  3. springboot集成shiro——使用RequiresPermissions注解无效

    在Springboot环境中继承Shiro时,使用注解@RequiresPermissions时无效 @RequestMapping("add") @RequiresPermiss ...

  4. Spring 学习笔记(五)—— Bean之间的关系、作用域、自动装配

    继承 Spring提供了配置信息的继承机制,可以通过为<bean>元素指定parent值重用已有的<bean>元素的配置信息. <?xml version="1 ...

  5. 【转】通过制作Flappy Bird了解Native 2D中的RigidBody2D和Collider

    作者:王选易,出处:http://www.cnblogs.com/neverdie/ 欢迎转载,也请保留这段声明.如果你喜欢这篇文章,请点[推荐].谢谢! 引子 在第一篇文章[Unity3D基础教程] ...

  6. 【转】Itween 贝塞尔曲线(一)

    原地址:点击打开链接 1.回调函数 回调函数,即当动画完成时那瞬间需要执行一次的一个函数,it中默认有一个onComplete函数,当动画完成时会自动执行,且你可以提供需要传递的一些参数.见如下代码: ...

  7. PHP异常处理类(文件上传提示)

    知识点: 大部分时候我们的代码总有各种各样的bug,新手程序员(比如我)最经常的工作就是不停的报错和echo变量,一个好的异常处理类可以帮我们更快+更容易理解报错代码的问题,同时,异常处理还可以避免一 ...

  8. display 垂直居中

    /* Center slide text vertically */ display: -webkit-box; display: -ms-flexbox; display: -webkit-flex ...

  9. mac 安装 navicat premium11.2.1400 破解版

    前言: 在 Mac 系统下 安装 navicat for mysql 时,遇到注册码的问题,适用了很多破解下载,都没有成功,现找到一篇特此记录下. Mac 下一款实用的数据库管理工具--Navicat ...

  10. 最全面的Android Webview详解

    转自:最全面的Android Webview详解 前言 现在很多App里都内置了Web网页(Hyprid App),比如说很多电商平台,淘宝.京东.聚划算等等,如下图  那么这种该如何实现呢?其实这是 ...