poj2406

题意

给出一个字符串,它是某个子串重复出现得到的,求子串最多出现的次数。

分析

后缀数组做的话得换上 DC3 算法。

那么子串的长度就是 \(len - height[rnk[0]]\) (当然必须保证字符串总长是子串长度的整数倍)。

如果字符串是 ababab,考虑 \(height[rnk[0]]\) 的意义,那么就是ababab和它的后缀排序后前面一个的后缀串的最大公共前缀长度,它前面的是abab,因为这个后缀串在比较的时候是以abab00的形式出现的,它后面接的只能是ababab,那么缺少的正好是一个子串。

code

  1. #include <cstdio>
  2. #include <cstring>
  3. #include <algorithm>
  4. #define F(x) ((x) / 3 + ((x) % 3 == 1 ? 0 : tb))
  5. #define G(x) ((x) < tb ? (x) * 3 + 1 : ((x) - tb) * 3 + 2)
  6. using namespace std;
  7. const int N = 1000005;
  8. int wa[N], wb[N], ws[N], wv[N], sa[N * 3];
  9. int rnk[N * 3], height[N * 3], s[N];
  10. char str[N];
  11. int c0(int *r, int a, int b) {
  12. return r[a] == r[b] && r[a + 1] == r[b + 1] && r[a + 2] == r[b + 2];
  13. }
  14. int c12(int k, int *r, int a, int b) {
  15. if (k == 2)
  16. return r[a] < r[b] || r[a] == r[b] && c12(1, r, a + 1, b + 1);
  17. return r[a] < r[b] || r[a] == r[b] && wv[a + 1] < wv[b + 1];
  18. }
  19. void Rsort(int *r, int *a, int *b, int n, int m) {
  20. for (int i = 0; i < n; i++) wv[i] = r[a[i]];
  21. for (int i = 0; i < m; i++) ws[i] = 0;
  22. for (int i = 0; i < n; i++) ws[wv[i]]++;
  23. for (int i = 1; i < m; i++) ws[i] += ws[i - 1];
  24. for (int i = n - 1; i >= 0; i--) b[--ws[wv[i]]] = a[i];
  25. }
  26. void dc3(int *r, int *sa, int n, int m) {
  27. int i, j, *rn = r + n, *san = sa + n, ta = 0, tb = (n + 1) / 3, tbc = 0, p;
  28. r[n] = r[n + 1] = 0;
  29. for (i = 0; i < n; i++) if (i % 3 != 0) wa[tbc++] = i;
  30. Rsort(r + 2, wa, wb, tbc, m);
  31. Rsort(r + 1, wb, wa, tbc, m);
  32. Rsort(r, wa, wb, tbc, m);
  33. for (p = 1, rn[F(wb[0])] = 0, i = 1; i < tbc; i++)
  34. rn[F(wb[i])] = c0(r, wb[i - 1], wb[i]) ? p - 1 : p++;
  35. if (p < tbc) dc3(rn, san, tbc, p);
  36. else for (i = 0; i < tbc; i++) san[rn[i]] = i;
  37. for (i = 0; i < tbc; i++) if (san[i] < tb) wb[ta++] = san[i] * 3;
  38. if (n % 3 == 1) wb[ta++] = n - 1;
  39. Rsort(r, wb, wa, ta, m);
  40. for (i = 0; i < tbc; i++) wv[wb[i] = G(san[i])] = i;
  41. for (i = 0, j = 0, p = 0; i < ta && j < tbc; p++)
  42. sa[p] = c12(wb[j] % 3, r, wa[i], wb[j]) ? wa[i++] : wb[j++];
  43. for (; i < ta; p++) sa[p] = wa[i++];
  44. for (; j < tbc; p++) sa[p] = wb[j++];
  45. }
  46. void calheight(int *r, int *sa, int n) {
  47. int i, j, k = 0;
  48. for (i = 1; i <= n; i++) rnk[sa[i]] = i;
  49. for (i = 0; i < n; height[rnk[i++]] = k)
  50. for (k ? k-- : 0, j = sa[rnk[i] - 1]; r[i + k] == r[j + k]; k++);
  51. }
  52. int main() {
  53. while (scanf("%s", str) == 1 && str[0] != '.') {
  54. int len = strlen(str);
  55. for (int i = 0; i < len; i++)
  56. s[i] = str[i] - 'a' + 1;
  57. s[len] = 0;
  58. dc3(s, sa, len + 1, 105);
  59. calheight(s, sa, len);
  60. /* 如果不懂打印出来看一下
  61. for(int i = 1; i <= len; i++) {
  62. printf("%s\n", str + sa[i]);
  63. }
  64. */
  65. int aa = len - height[rnk[0]];
  66. int ans = 1;
  67. if(len % aa == 0) {
  68. ans = len / aa;
  69. }
  70. printf("%d\n", ans);
  71. }
  72. return 0;
  73. }

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

  1. POJ2406 Power Strings —— KMP or 后缀数组 最小循环节

    题目链接:https://vjudge.net/problem/POJ-2406 Power Strings Time Limit: 3000MS   Memory Limit: 65536K Tot ...

  2. 【POJ2406】Power Strings(KMP,后缀数组)

    题意: n<=1000000,cas较大 思路:这是一道论文题 后缀数组已弃疗,强行需要DC3构造,懒得(不会)写 ..]of longint; n,m,i,j,len,ans,st:longi ...

  3. POJ2406 Power Strings(KMP,后缀数组)

    这题可以用后缀数组,KMP方法做 后缀数组做法开始想不出来,看的题解,方法是枚举串长len的约数k,看lcp(suffix(0), suffix(k))的长度是否为n- k ,若为真则len / k即 ...

  4. POJ2406 Power Strings 【KMP 或 后缀数组】

    电源串 时间限制: 3000MS   内存限制: 65536K 提交总数: 53037   接受: 22108 描述 给定两个字符串a和b,我们定义a * b是它们的连接.例如,如果a =" ...

  5. 【距离GDOI:131天】 后缀数组完毕

    用了近两周的时间,终于把罗神那篇后缀数组应用看完了,题目也写了一遍,T了无数次...详见前几篇博文... 后缀数组很重要的是那个height数组,可以用来做各种奇奇怪怪的东西...常用方法去是去二分, ...

  6. 后缀数组基本问题QAQ

    以下题目均来自罗穗骞的论文... No.1最长公共前缀 最长公共前缀: 题目: 给定一个字符串,询问某两个后缀的最长公共前缀. 分析: 某两个后缀的最长公共前缀就是区间height最小值,转化为RMQ ...

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

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

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

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

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

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

随机推荐

  1. 【Regularization】林轩田机器学习基石

    正则化的提出,是因为要解决overfitting的问题. 以Linear Regression为例:低次多项式拟合的效果可能会好于高次多项式拟合的效果. 这里回顾上上节nonlinear transf ...

  2. 洛谷P1101单词方阵

    题目描述 给一n×n的字母方阵,内可能蕴含多个“yizhong”单词.单词在方阵中是沿着同一方向连续摆放的. 摆放可沿着 8个方向的任一方向,同一单词摆放时不再改变方向,单词与单词之间可以交叉,因此有 ...

  3. leetcode 179. 最大数 解题报告

    给定一组非负整数,重新排列它们的顺序使之组成一个最大的整数. 示例 1: 输入: [10,2] 输出: 210 示例 2: 输入: [3,30,34,5,9] 输出: 9534330 说明: 输出结果 ...

  4. URAL 1732. Ministry of Truth ( KMP 多模式串匹配 )

    问在第一个串中删掉几个字符能否得到第二个串.注意在第二个串中不连续的单词在第一个串中也必须不连续. 一组数据: Input: abababbbbababbb aba ab Output: I HAVE ...

  5. canvas 基础

    1.<canvas>元素 <canvas id="tutorial" width="150" height="150"&g ...

  6. 【bzoj1270】[BeijingWc2008]雷涛的小猫 dp

    题目描述   输入 输出 样例输入 样例输出 8 题解 dp 设f[i][j]表示在第i棵树的j高度时最多吃到的柿子数. 那么只有两种可能能够到达这个位置:滑下来.跳下来. 滑下来直接用f[i][j+ ...

  7. CF995E Number Clicker 解题报告

    CF995E Number Clicker 题目描述 Allen is playing Number Clicker on his phone. He starts with an integer u ...

  8. uoj228:基础数据结构练习题

    题意:http://uoj.ac/problem/228 sol  :线段树开根操作 对于节点x,可以在max[x]-min[x]<=1时直接做,转化为区间减或区间覆盖 #include< ...

  9. BZOJ[Sdoi2014]数表 莫比乌斯反演

    [Sdoi2014]数表 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 2383  Solved: 1229[Submit][Status][Disc ...

  10. TortoiseSVN里checkout depth各选项的含义

    代表四种检出深度: 1.Fully recursive——全递归:检出完整的目录树,包含所有的文件或子目录.2.Immediate children,including folders——直接子节点, ...