标签:扩展kmp
链接:http://acm.hdu.edu.cn/showproblem.php?pid=4333
题意:给以数字字符串,移动最后若干位到最前边,统计得到的数字有多少比原来大,有多少和原来相同,有多少比原来的小。
思路:拓展KMP中的next数组标记的是子串和母串的公共前缀的长度,要将字符串长度变成原来二倍,这样如果变换后不是完全相同的数字也即公共前缀长度大于等于字符串长度,那么字母串公共前缀的下一位的大小比较就是题目所要求的比较。由于相同的数字串只算一次,则只要统计比较第一个“循环节”里的改变数字串即可。
资料:【转】扩展KMP http://wenku.baidu.com/view/8e9ebefb0242a8956bece4b3.html
#include <iostream>
#include <cstring>
#include <cstdio>
using namespace std;
int ext[1000005*2];
char ss[1000005*2];
void getnext(char *T,int *next)
{
int i,length = strlen(T);
next[0] = length;
for(i = 0; i<length-1 && T[i]==T[i+1]; i++);
next[1] = i;
int a = 1;
for(int k = 2; k < length; k++)
{
int p = a+next[a]-1, L = next[k-a];
if( (k-1)+L >= p )
{
int j = (p-k+1)>0? (p-k+1) : 0;
while(k+j<length && T[k+j]==T[j]) j++;
next[k] = j, a = k;
}
else next[k] = L;
}
}
int main()
{
int tot,a,b,c;
scanf("%d",&tot);
for(int ii=1; ii<=tot; ii++)
{
a=b=c=0;
scanf("%s",ss);
int len=strlen(ss);
for(int i=len; i<len*2; i++)
ss[i]=ss[i-len];
getnext(ss,ext);
ss[2*len]='\0';
int k=len;
for(int i=1;i<=len;i++)
{
if(i+ext[i]>=len)
{
k=len%i?len:i;
break;
}
}
for(int i=0; i<k; i++)
{
if(ext[i]>=len)
b++;
else if(ss[i+ext[i]]>ss[ext[i]])
a++;
else c++;
}
printf("Case %d: %d %d %d\n",ii,c,b,a);
}
return 0;
}
HDU 4333 Revolving Digits 扩展KMP,布布扣,bubuko.com
HDU 4333 Revolving Digits 扩展KMP
标签:扩展kmp
原文地址:http://blog.csdn.net/ooooooooe/article/details/36712115