Given two strings a and b we define a*b to be their concatenation. For example, if a = "abc" and b = "def" then a*b = "abcdef". If we think of concatenation as multiplication, exponentiation by a non-negative integer is defined in the normal way: a^0 = "" (the empty string) and a^(n+1) = a*(a^n).

Input

Each test case is a line of input representing s, a string of printable characters. The length of s will be at least 1 and will not exceed 1 million characters. A line containing a period follows the last test case.

Output

For each s you should print the largest n such that s = a^n for some string a.

Sample Input

abcd
aaaa
ababab
.

Sample Output

1
4
3

Hint

This problem has huge input, use scanf instead of cin to avoid time limit exceed.
 
 
 
题意:
输入一个字符串,让你找出来这个串的最小循环节
 
 
题解:
对于答案就用for循环从1到n枚举(为什么不能用二分?因为这个判断没有单调性。即i不是循环节的话,你不能说i-1或者i+1也肯定不是循环节)
 
下一步就是判断枚举的这个i是不是循环节,怎么判断?
满足这三个条件:n%i==0 && Rank[0]-Rank[i]==1 && height[Rank[0]]==n-i
 
1、n%i==0这就肯定不需要说了
2、你要注意Rank[0]表示的是啥,他代表后缀[0...n]的排名是多少,如果循环节是i,那么Rank[0]-Rank[i]==Rank[i]-Rank[2*i]......==1
3、如果循环节是i,那么height[Rank[0]]==n-i,   height[Rank[0]]代表后缀[0..n]与后缀[i...n]的最长公共前缀,既然循环节是i,那么height[Rank[0]]的值也就等于n-i
 
还要注意这道题用后缀数组来做的话要用dc3模板,如果用Da模板会TLE
 
dc3模板AC代码:
 1 #include <cstdlib>
2 #include <cstring>
3 #include <cstdio>
4 #include <algorithm>
5 using namespace std;
6 const int maxn = 3000010;
7 #define F(x) ((x) / 3 + ((x) % 3 == 1 ? 0 : tb))
8 #define G(x) ((x) < tb ? (x) * 3 + 1 : ((x) - tb) * 3 + 2)
9 int wa[maxn], wb[maxn], Ws[maxn], wv[maxn], sa[maxn];
10 int Rank[maxn], height[maxn],r[maxn];
11 char s[maxn];
12 int c0(int *r, int a, int b)
13 {
14 return r[a] == r[b] && r[a + 1] == r[b + 1] && r[a + 2] == r[b + 2];
15 }
16 int c12(int k, int *r, int a, int b)
17 {
18 if (k == 2)
19 return r[a] < r[b] || r[a] == r[b] && c12(1, r, a + 1, b + 1);
20 return r[a] < r[b] || r[a] == r[b] && wv[a + 1] < wv[b + 1];
21 }
22 void Rsort(int *r, int *a, int *b, int n, int m)
23 {
24 for (int i = 0; i < n; i++) wv[i] = r[a[i]];
25 for (int i = 0; i < m; i++) Ws[i] = 0;
26 for (int i = 0; i < n; i++) Ws[wv[i]]++;
27 for (int i = 1; i < m; i++) Ws[i] += Ws[i - 1];
28 for (int i = n - 1; i >= 0; i--) b[--Ws[wv[i]]] = a[i];
29 }
30 void dc3(int *r,int *sa,int n, int m)
31 {
32 int i, j, *rn = r + n, *san = sa + n, ta = 0, tb = (n + 1) / 3, tbc = 0, p;
33 r[n] = r[n + 1] = 0;
34 for (i = 0; i < n; i++) if (i % 3 != 0) wa[tbc++] = i;
35 Rsort(r + 2, wa, wb, tbc, m);
36 Rsort(r + 1, wb, wa, tbc, m);
37 Rsort(r, wa, wb, tbc, m);
38 for (p = 1, rn[F(wb[0])] = 0, i = 1; i < tbc; i++)
39 rn[F(wb[i])] = c0(r, wb[i - 1], wb[i]) ? p - 1 : p++;
40 if (p < tbc) dc3(rn, san, tbc, p);
41 else for (i = 0; i < tbc; i++) san[rn[i]] = i;
42 for (i = 0; i < tbc; i++) if (san[i] < tb) wb[ta++] = san[i] * 3;
43 if (n % 3 == 1) wb[ta++] = n - 1;
44 Rsort(r, wb, wa, ta, m);
45 for (i = 0; i < tbc; i++) wv[wb[i] = G(san[i])] = i;
46 for (i = 0, j = 0, p = 0; i < ta && j < tbc; p++)
47 sa[p] = c12(wb[j] % 3, r, wa[i], wb[j]) ? wa[i++] : wb[j++];
48 for (; i < ta; p++) sa[p] = wa[i++];
49 for (; j < tbc; p++) sa[p] = wb[j++];
50 }
51 void get_height(int n)
52 {
53 int i, j, k = 0;
54 for (i = 1; i <= n; i++) Rank[sa[i]] = i;
55 for (i = 0; i < n; height[Rank[i++]] = k)
56 for (k ? k-- : 0, j = sa[Rank[i] - 1]; r[i + k] == r[j + k]; k++);
57 }
58 int main()
59 {
60 while(~scanf("%s",s))
61 {
62 int n=strlen(s);
63 if(n==1 && s[0]=='.') break;
64 for(int i=0;i<n;++i)
65 r[i]=s[i];
66 r[n]='0';
67 dc3(r,sa,n+1,200);
68 get_height(n);
69 int flag=0;
70 for(int i=1;i<n;++i) //枚举循环节长度
71 {
72 if(n%i==0 && Rank[0]-Rank[i]==1 && height[Rank[0]]==n-i)
73 {
74 flag=1;
75 printf("%d\n",n/i);
76 break;
77 }
78 }
79 if(!flag)
80 printf("1\n");
81 }
82 return 0;
83 }

Da模板TLE代码:

 1 #include <cstdlib>
2 #include <cstring>
3 #include <cstdio>
4 #include <algorithm>
5 using namespace std;
6
7 const int N = 1000010;
8 int x[N], y[N], c[N];
9 int rank[N], height[N];
10 int sa[N],n,k;
11 char s[N];
12 bool pan(int *x,int i,int j,int k,int n)
13 {
14 int ti=i+k<n?x[i+k]:-1;
15 int tj=j+k<n?x[j+k]:-1;
16 return x[i]==x[j]&&ti==tj;
17 }
18 void build_SA(int n,int r)
19 {
20 int *x=rank,*y=height;
21 for(int i=0; i<r; i++)c[i]=0;
22 for(int i=0; i<n; i++)c[s[i]]++;
23 for(int i=1; i<r; i++)c[i]+=c[i-1];
24 for(int i=n-1; i>=0; i--)sa[--c[s[i]]]=i;
25 r=1;
26 x[sa[0]]=0;
27 for(int i=1; i<n; i++)
28 x[sa[i]]=s[sa[i]]==s[sa[i-1]]?r-1:r++;
29 for(int k=1; r<n; k<<=1)
30 {
31 int yn=0;
32 for(int i=n-k; i<n; i++)y[yn++]=i;
33 for(int i=0; i<n; i++)
34 if(sa[i]>=k)y[yn++]=sa[i]-k;
35 for(int i=0; i<r; i++)c[i]=0;
36 for(int i=0; i<n; i++)++c[x[y[i]]];
37 for(int i=1; i<r; i++)c[i]+=c[i-1];
38 for(int i=n-1; i>=0; i--)sa[--c[x[y[i]]]]=y[i];
39 swap(x,y);
40 r=1;
41 x[sa[0]]=0;
42 for(int i=1; i<n; i++)
43 x[sa[i]]=pan(y,sa[i],sa[i-1],k,n)?r-1:r++;
44 }
45 for(int i=0; i<n; i++)rank[i]=x[i];
46 }
47 void get_height(int n)
48 {
49 int i,j,k=0;
50 for(i=1; i<=n; i++)rank[sa[i]]=i;
51 for(i=0; i<n; i++)
52 {
53 if(k)k--;
54 else k=0;
55 j=sa[rank[i]-1];
56 while(s[i+k]==s[j+k])k++;
57 height[rank[i]]=k;
58 }
59 }
60 int check(int len)
61 {
62 int i=2,cnt=0;
63 while(1)
64 {
65 while(i<=n && height[i]>=len)
66 cnt++,i++;
67 if(cnt+1>=k)return 1;
68 if(i>=n)return 0;
69 while(i <=n &&height[i]<len)
70 i++;
71 cnt=0;
72 }
73 }
74
75 int main()
76 {
77 while(~scanf("%s",s))
78 {
79 n=strlen(s);
80 if(n==1 && s[0]=='.') break;
81 s[n]='0';
82 build_SA(n+1,200);
83 get_height(n);
84 //printf("%d %d %d\n",rank[0],rank[3],height[rank[0]]);
85 int flag=0;
86 for(int i=1;i<n;++i) //枚举循环节长度
87 {
88 if(n%i==0 && rank[0]-rank[i]==1 && height[rank[0]]==n-i)
89 {
90 flag=1;
91 printf("%d\n",n/i);
92 break;
93 }
94 }
95 if(!flag)
96 printf("1\n");
97 }
98 return 0;
99 }

Power Strings POJ - 2406 后缀数组的更多相关文章

  1. Power Strings POJ - 2406

    Power Strings POJ - 2406 时限: 3000MS   内存: 65536KB   64位IO格式: %I64d & %I64u 提交 状态 已开启划词翻译 问题描述 Gi ...

  2. Power Strings POJ - 2406(next水的一发 || 后缀数组)

    后缀数组专题的 emm.. 就next 循环节../ 有后缀数组也可以做 从小到大枚举长度i,如果长度i的子串刚好是重复了len/i次,应该满足len % i == 0和rank[0] - rank[ ...

  3. ( KMP 求循环节的个数)Power Strings -- poj -- 2406

    链接: http://poj.org/problem?id=2406 Power Strings Time Limit:3000MS     Memory Limit:65536KB     64bi ...

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

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

  5. Power Strings (poj 2406 KMP)

    Language: Default Power Strings Time Limit: 3000MS   Memory Limit: 65536K Total Submissions: 33205   ...

  6. Match:Power Strings(POJ 2406)

     字符串前缀的阶 题目大意:求前缀的阶 和POJ1961是一样的,KMP的Next数组的应用,不要用STL,不要一个一个读入字符(IO永远是最慢的) #include <iostream> ...

  7. Power Strings - POJ 2406(求循环节)

    题目大意:叙述的比较高大上,其实就是一个字符串B = AAAAAAA,求出来这个A最短有多长   分析:注意如果这个串不是完全循环的,那么循环节就是就是它本身.   代码如下: #include< ...

  8. poj 3693 后缀数组 重复次数最多的连续重复子串

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

  9. poj 2406 Power Strings (kmp 中 next 数组的应用||后缀数组)

    http://poj.org/problem?id=2406 Power Strings Time Limit: 3000MS   Memory Limit: 65536K Total Submiss ...

随机推荐

  1. 温故而知新--day2

    温故而知新--day2 类 类与对象 类是一个抽象的概念,是指对现实生活中一类具有共同特征的事物的抽象.其实列化后称为对象.类里面由类属性组成,类属性可以分为数据属性和函数属性(函数属性又称为类方法) ...

  2. VmwareTools显示灰色无法安装

    VMware不安装VMware Tools无法全屏,然后实机之间不能传输文件等. 安装Vmware Tools显示是灰色的,详细解决方案如下 打开虚拟机设置,CD/DVD 选择ISO映像文件 在Vmw ...

  3. RPC 是通信协议吗 ?→ 我们来看下它的演进过程

    开心一刻 一实习小护士给我挂针,拿着针在我胳膊上扎了好几针也没找到血管 但这位小姑娘真镇定啊,表情严肃认真,势有不扎到血管不罢休的意思 十几针之后,我忍着剧痛,带着敬畏的表情问小护士:你这针法跟容嬷嬷 ...

  4. 在recover database时,如何决定该从哪一个SCN开始恢复

    使用备份恢复的方法搭建DG库,还原数据文件后,打开数据库时报错 SQL> ALTER DATABASE OPEN READ ONLY; ALTER DATABASE OPEN READ ONLY ...

  5. EntityFramework Core如何映射动态模型?

    前言 本文我们来探讨下映射动态模型的几种方式,相信一部分童鞋项目有这样的需求,比如每天/每小时等生成一张表,此种动态模型映射非常常见,经我摸索,这里给出每一步详细思路,希望能帮助到没有任何头绪的童鞋, ...

  6. Mysql 中写操作时保驾护航的三兄弟!

    这期的文章主要是讲述写操作过程中涉及到的三个日志文件,看过前几期的话可能你或多或少已经有些了解了(或者从别的地方也了解过).比如整个写操作过程中用到的两阶段提交,又或者是操作过程中涉及到的日志文件,但 ...

  7. linux GPU上多个buffer间的同步 —— ww_mutex、dma-fence的使用 笔记

    原文链接:https://www.cnblogs.com/yaongtime/p/14111134.html   WW-Mutexes   在GPU中一次Render可能会涉及到对多个buffer的引 ...

  8. Ubuntu14.04系统安装

    1. 使用U盘或光盘进行引导进入系统安装向导. 2. 安装类型选择,选择中文(简体).然后点安装ubuntu. 3. 安装ubuntu电脑必须接入外网(外网的方式有自动获取或手动编辑IP地址). 网络 ...

  9. 对于两个输入文件,即文件A 和文件B ,请编写MapReduce程序,对两个文件进行合并排除其中重复的内容,得到一个新的输出文件C。

    package org.apache.hadoop.examples; import java.util.HashMap; import java.io.IOException; import jav ...

  10. RVA与FOA的转换

    主要取决于文件对齐与内存对齐的值