POJ1961 子串重复出现最大次数

给定一个字符串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;
}


你可能感兴趣的:(POJ1961 子串重复出现最大次数)