题目大意:
白书例题
给出一个长度不超过1000000的字符串S, 对于该字符串的所有前缀求其周期, 如果周期K >= 2输出起始位置是第几个字符和其周期K
每一个Test case之后都要有一个空行
大致思路:
就是利用KMP的next数组的性质对于长度为n的字符串如果n % (n – next[n]) == 0则是周期串, 周期的部分长度为n – next[n], 周期数为 n / (n – next[n])
从头到尾扫描一次next数组即可, 时间复杂度O(|S|)
代码如下:
Result : Accepted Memory : ? KB Time : 49 ms
/*
* Author: Gatevin
* Created Time: 2015/2/11 15:47:42
* File Name: Mononobe_Mitsuki.cpp
*/
#include<iostream>
#include<sstream>
#include<fstream>
#include<vector>
#include<list>
#include<deque>
#include<queue>
#include<stack>
#include<map>
#include<set>
#include<bitset>
#include<algorithm>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cctype>
#include<cmath>
#include<ctime>
#include<iomanip>
using namespace std;
const double eps(1e-8);
typedef long long lint;
char in[1000010];
int next[1000010];
int n;
void KMP()
{
memset(next, 0, sizeof(next));
for(int i = 1; i < n; i++)//动态规划求next数组
{
int j = i;
while(j != 0)
{
j = next[j];
if(in[j] == in[i])
{
next[i + 1] = j + 1;
break;
}
}
}
for(int i = 1; i <= n; i++)//依次判断周期
if(i % (i - next[i]) == 0 && i /(i - next[i]) >= 2)
printf("%d %d\n", i, i / (i - next[i]));
printf("\n");
}
int main()
{
int cas = 1;
while(scanf("%d", &n), n)
{
scanf("%s", in);
printf("Test case #%d\n", cas++);
KMP();
}
return 0;
}