给定一个字符串s,设它p为它的前缀,若多个p(大于等于2个)相连接之后仍为s的某个前缀,则输出相关信息。
如:s="abcabcabcab",则对于p="abc",两个p相连接之后为“abcabc”,三个p连接之后为"abcabcabc",这两者都是原串s的前缀,则输出多个p连接之后的前缀长度,以及它是由多少个p连接而成的。若对于某个满足条件的前缀,它既可以由n1个p1连接而成,又可以由n2个p2连接而成,则输出n1和n2中较大者的那种情况。
最后输出的答案要求多个p连接之后的前缀长度是递增的。
对于s中的某个位置i - 1,此处(i >= 1),若i % (i - next[i]) == 0则原串必定满足是由1个或多个p连接而成的,所以加上i / (i - next[i]) > 1就可以过滤掉只由一个p组成的情况。此处的next就是kmp算法的回滚数组。因此只要算出next数组,然后再从左往右扫描一次就可以了。
#include <iostream>
#include <cstdio>
#include <cstring>
using namespace std;
int Next[1000005];
char str[1000005];
int len;
void getNext()
{
Next[0] = -1;
int i = 0, j = -1;
while (i < len)
{
if (j == -1 || str[i] == str[j])
{
++i;
++j;
Next[i] = j;
}
else j = Next[j];
}
}
int main()
{
int t = 1;
while (scanf("%d", &len) != EOF && len)
{
scanf("%s", str);
getNext();
printf("Test case #%d\n", t++);
for (int i = 2; i <= len; ++i)
{
if (i % (i - Next[i]) == 0 && i / (i - Next[i]) > 1)
{
printf("%d %d\n", i, i / (i - Next[i]));
}
}
printf("\n");
}
return 0;
}