Description
“Madoka,不要相信 QB!”伴随着 Homura 的失望地喊叫,Madoka 与 QB 签订了契约.
这是 Modoka 的一个噩梦,也同时是上个轮回中所发生的事.为了使这一次 Madoka 不再与 QB签订契约,Homura 决定在刚到学校的第一天就解决 QB.然而,QB 也是有许多替身的(但在第八话中的剧情显示它也有可能是无限重生的),不过,意志坚定的 Homura 是不会放弃的——她决定
消灭所有可能是 QB 的东西.现在,她已感受到附近的状态,并且把它转化为一个长度为 n 的字符串交给了学 OI 的你.
现在你从她的话中知道 , 所有形似于 A+B+A 的字串都是 QB 或它的替身 , 且len(A)>=k,len(B)>=1 (位置不同其他性质相同的子串算不同子串,位置相同但拆分不同的子串算同一子串),然后你必须尽快告诉 Homura 这个答案——QB 以及它的替身的数量.
Input
第一行一个字符串,第二行一个数 k
Output
仅一行一个数 ans,表示 QB 以及它的替身的数量
Sample Input
【样例输入 1】
aaaaa
1
【样例输入 2】
abcabcabc
2
aaaaa
1
【样例输入 2】
abcabcabc
2
Sample Output
【样例输出 1】
6
6
【样例输出 2】
8
解题思路:
首先一上来想到的是:给你一个字符串,怎么快速地判断是不是ABA式的。
可以看出对于字符串匹配的算法(hash/KMP)
hash在这个问题上是远远劣于KMP的:
hash的暴力枚举时间复杂度已经不可承受,而且似乎没有方法优化的样子。
KMP有一个优秀的性质:你可以很快的知道一个字符串前缀后缀重复的情况,而这恰恰就是ABA式的字符串。
简单来说假如说给定起点字符串的起点KMP求next数组的时候判断其小于k的最小前缀,存在则ans++
那么每次暴力枚举字符串起始点进行如上算法,输出答案就好了,注意前缀的next也是合法情况,开数组记录最小值就好了。
时间复杂度O(n/2),强调/2是有原因的n<=15000
代码:
#include<cstdio>
#include<cstring>
#include<algorithm>
using std::min;
char ch[];
int nxt[];
int mini[];
int ans;
int n,m,k;
void Kmp(char *a)
{
nxt[]=nxt[]=;
mini[]=0x3f3f3f3f;
int len=strlen(a+);
for(int i=,j=;i<=len;i++)
{
while(j&&a[i]!=a[j+])
j=nxt[j];
if(a[i]==a[j+])
j++;
nxt[i]=j;
if(j<k)
mini[j]=0x3f3f3f3f;
else
mini[j]=min(mini[nxt[j]],j);
if(mini[j]*<i)
ans++;
}
}
int main()
{
scanf("%s",ch+);
scanf("%d",&k);
int lenl=strlen(ch+);
for(int i=;i<lenl;i++)
Kmp(ch+i);
printf("%d\n",ans);
return ;
}