poj 3641 ——2016——3——15

时间:2022-04-12 20:22:02

传送门:http://poj.org/problem?id=3461

题目大意:给你两个字符串p和s,求出p在s中出现的次数。

题解:这一眼看过去就知道是KMP,作为模板来写是最好不过了。。。。

这道题我写了两种风格的kmp,个人感觉第2种好理解一些;

 #include <iostream>
#include <cstring>
#include <cstdio>
#define inf
char s[],p[];
int next[];
int lens,lenp,n;
using namespace std;
void getnext()
{
int j=-,i=;
next[]=-;
while (i!=lenp)
{
if (j==- || s[i]==s[j])
next[++i]=++j;
else
j=next[j];
}
}
int KMP()
{
int i=,j=,count=;
while (i!=lens && j!=lenp)
{
if (s[i]==p[j]||j==-)
++i,++j;
else
j=next[j];
if (j==lenp)
{
count++;
j=next[j];
}
}
return count;
}
int main()
{
int z;
scanf("%d",&z);
for (int zz=; zz<=z; zz++)
{
scanf("%s%s",p,s);
lens=strlen(s);
lenp=strlen(p);
//memset(next,0,sizeof(next));
getnext();
int ans=KMP();
printf("%d\n",ans);
}
}

第一种

 #include<iostream>
#include<cstring>
#include<cstdio>
#define maxn 1000100
int n,fix,ans,i,lens,lent;
char s[maxn],t[maxn];
int next[maxn];
void getnext()
{
fix=;
for (fix,i=; i<=lent; i++)
{
while(fix && t[fix+]!=t[i]) fix=next[fix];
if (t[fix+]==t[i]) fix++;
next[i]=fix;
}
}
int KMP()
{
int count;
fix=; count=;
for (int i=; i<=lens; i++)
{
while (fix && t[fix+]!=s[i]) fix=next[fix];
if (t[fix+]==s[i]) fix++;
if (fix==lent)
{
count++;
fix=next[fix];
}
}
return count;
}
int main()
{
int z;
scanf("%d",&z);
for (int zz=; zz<=z; zz++)
{
scanf("%s",t+);
scanf("%s",s+);
lens=strlen(s+); lent=strlen(t+);
memset(next,,sizeof(next));
getnext();
ans=KMP();
printf("%d\n",ans);
}
}

第二种