1、传送门:http://begin.lydsy.com/JudgeOnline/problem.php?id=2725

  题目大意:找一个串在另一个串中出现的次数

  题解:kmp(纯裸题)

  

 #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 (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);
}
return ;
}

  

2、传送门:http://begin.lydsy.com/JudgeOnline/problem.php?id=2732

  题目大意:给你一个字符串,让你求出最大重复周期(最大周期不和本身重合)

  题解:kmp或者扩展kmp(但我并未用这种方法),我用的是kmp,但是一直WA,原来是求next数组把while写成了if(手抖毁一生)。

     好吧,写题解了:用kmp求出next数组,然后在去递归next[n],因为j=next[next[n]]一直next下去直到其的next为0就ans+=n-j;

     这就可以求出不和母串一样的最大重复周期,但是这又有一个问题你在求j时要递归时间就有可能为n^2的所以在每次递归时改变next的值就好了(详见代码);

  

 #include<iostream>
#include<cstring>
#include<cstdio>
#define inf 0x7ffffff
#define maxn 1000100
char s[maxn];
int nnext[maxn];
int n,i,j,fix;
long long ans;
using namespace std;
int main()
{
scanf("%d",&n);
scanf("%s",s+);
fix=;
for (int i=; i<=n; i++)
{
while (fix && s[fix+]!=s[i]) fix=nnext[fix];
if (s[fix+]==s[i]) fix++;
nnext[i]=fix;
}
ans=;
for (int i=; i<=n; i++)
{
int j=i;
if (!nnext[j]) continue;
while (nnext[nnext[j]]) nnext[j]=nnext[nnext[j]];
ans+=(i-nnext[j]);
}
printf("%lld\n",ans);
}

3、传送门:http://begin.lydsy.com/JudgeOnline/problem.php?id=2726

  题目大意:给出一个字母组成的矩阵,找出一个最小的子矩阵,使得这个子矩阵的无限复制扩张之后的矩阵包含原来矩阵如:         

        ABABA

        ABABA 
        他的最小重复子矩阵是AB

  题解:还是kmp,只不过要有一点小技巧,就是把一行当作一个元素,那么就有n个元素,然后做kmp找重复子串,那么最小重复子串就为n-next[n],列也做此操作,答案就是

      (n-r[n])*(m-c[m]);

  

 #include <cstdio>
#include <cstring>
char s[][], rs[][];
int R[], C[];
int r, c; void get_nextR()
{
R[] = -;
int j = -, i = ;
while(i < r)
{
if(j == - || strcmp(s[i], s[j]) == )
{
i++;
j++;
R[i] = j;
}
else
j = R[j];
}
} void get_nextC()
{
C[] = -;
int j = -, i = ;
while(i < c)
{
if(j == - || strcmp(rs[i], rs[j]) == )
{
i++;
j++;
C[i] = j;
}
else
j = C[j];
}
} int main()
{
while(scanf("%d %d", &r, &c) != EOF)
{
for(int i = ; i < r; i++)
scanf("%s", s[i]);
get_nextR();
for(int i = ; i < r; i++)
for(int j = ; j < c; j++)
rs[j][i] = s[i][j];
get_nextC();
printf("%d\n", (r - R[r]) * (c - C[c]));
}
}

4、传送门:http://begin.lydsy.com/JudgeOnline/problem.php?id=2724

  题目大意:给定一个字符串,要求找到同时是它前缀也是后缀的字符串的个数,并且输出他们的长度。

  题解:理解一下next数组的用法,从next[n]一直往前求next,那么那些点的坐标就是answer。

 #include<iostream>
#include<cstring>
#include<cstdio>
#define maxn 1000100
int n,ans,fix;
int nnext[maxn];
char s[maxn];
using namespace std;
void show(int n)
{
if (n<=) return;
show(nnext[n]);
printf("%d ",n);
}
int main()
{
scanf("%d",&n);
scanf("%s",s+);
fix=;
for (int i=; i<=n; i++)
{
while (fix && s[fix+]!=s[i]) fix=nnext[fix];
if (s[fix+]==s[i]) fix++;
nnext[i]=fix;
}
show(n);
return ;
}

5、传送门:http://begin.lydsy.com/JudgeOnline/problem.php?id=2723

 题目大意:

    现在给一个字符串A,和另一个字符串B。要你每次从B串中从左到右第第一个A串。 
    并且从B串中删除它,直到A串不为B串的子串。问需要几次删除操作。

  题解:next数组的应用,只不过用一个堆记录那些没匹配成功的字符,在用它进行匹配。

 #include<cstring>
#include<cstdio>
#include<cstring>
char t[],s[];
int m,n,top,i,fix,ans;
int z[],next[],f[];
using namespace std;
int main()
{
scanf("%s%s",t+,s+);
m=strlen(s+); n=strlen(t+);
for (fix=,i=; i<=n; i++)
{
while (fix && t[fix+]!=t[i]) fix=next[fix];
if (t[fix+]==t[i]) fix++;
next[i]=fix;
}
for (int i=; i<=m; i++)
{
fix=f[z[top]];
while (fix && t[fix+]!=s[i]) fix=next[fix];
if (t[fix+]==s[i]) fix++; if (fix==n) top-=(n-),ans++;
else
f[i]=fix, z[++top]=i;
}
printf("%d\n",ans);
}

6、传送门:http://begin.lydsy.com/JudgeOnline/problem.php?id=2719

  题目大意:给出一个串S,它包含 N 个字符。设 Pi = S [ 1 .. I ] ,对于一些 Pi , 如果Pi能表示成K个字符串相连而成的(K > 1 ,且K尽量大),则打印I和K。

  题解:如果i mod(i-next[i]) =0 && (i/(i-next[i])>1 输出答案(i);

 #include<iostream>
#include<cstring>
#include<cstdio>
#define inf 0x7fffffff
int n,ans;
int nnext[];
char s[];
using namespace std;
int main()
{
int z=;
while (true)
{
scanf("%d",&n);
if (n==) break;
z++;
printf("Test case #%d\n",z);
scanf("%s",s+);
ans=;
int fix=;
for (int i=; i<=n; i++)
{
while (fix && s[fix+]!=s[i]) fix=nnext[fix];
if (s[fix+]==s[i]) fix++;
nnext[i]=fix;
}
for (int i=; i<=n; i++)
{
int k=i-nnext[i];
if (i % k== && i/k>) printf("%d %d\n",i,i/k);
}
printf("\n");
}
}

7、传送门:http://begin.lydsy.com/JudgeOnline/problem.php?id=2720

  题目大意:给你一个字符串,它是由某个字符串不断自我连接形成的。 但是这个字符串是不确定的,现在只想知道它的最短长度是多少.

  题解:同上;

 #include<iostream>
#include<cstring>
#include<cstdio>
#define inf 0x7fffffff
int n,ans;
int nnext[];
char s[];
using namespace std;
int main()
{
scanf("%d",&n);
scanf("%s",s+);
ans=;
int fix=;
for (int i=; i<=n; i++)
{
while (fix && s[fix+]!=s[i]) fix=nnext[fix];
if (s[fix+]==s[i]) fix++;
nnext[i]=fix;
}
printf("%d\n",n-nnext[n]);
}

2016——3——16 kmp 7题的更多相关文章

  1. mysql查询练习题-2016.12.16

    >>>>>>>>>> 练习时间:2016.12.16 编辑时间:2016-12-20-->22:12:08 题: 涉及:多表查询.ex ...

  2. 2016.8.16上午纪中初中部NOIP普及组比赛

    2016.8.16上午纪中初中部NOIP普及组比赛 链接:https://jzoj.net/junior/#contest/home/1334 这次也翻车了,感觉比之前难多了. 辛辛苦苦改完了,太难改 ...

  3. zstu.4194: 字符串匹配(kmp入门题&& 心得)

    4194: 字符串匹配 Time Limit: 1 Sec  Memory Limit: 128 MB Submit: 206  Solved: 78 Description 给你两个字符串A,B,请 ...

  4. 【转载】webstorm11(注册,激活,破解,码,一起支持正版,最新可用)(2016.11.16更新)

    很多人都发现 http://idea.lanyus.com/ 不能激活了 很多帖子说的 http://15.idea.lanyus.com/ 之类都用不了了 最近封的厉害仅作测试 选择 License ...

  5. HDU 1711 Number Sequence(KMP裸题,板子题,有坑点)

    Number Sequence Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) ...

  6. 51Nod 1277 字符串中的最大值(KMP,裸题)

    1277 字符串中的最大值 题目来源: Codility 基准时间限制:1 秒 空间限制:131072 KB 分值: 80 难度:5级算法题 一个字符串的前缀是指包含该字符第一个字母的连续子串,例如: ...

  7. poj-2406(kmp水题)

    题意:定义一个a*b=字符串a连接字符串b:给你一个字符串s,问你这个字符串最多能用多少个字符串t连接得到:例如:aaaa=4个a构成: 解题思路:kmp水题,next数组除了查找字串以外最广泛的一种 ...

  8. HDU 1711 - Number Sequence - [KMP模板题]

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1711 Time Limit: 10000/5000 MS (Java/Others) Memory L ...

  9. POJ 3167 Cow Pattern ★(KMP好题)

    题意 给你一个数字序列S,再给一个数字序列pattern,S和pattern中的数字都是1到s(s<=25).每个序列里的数字都有个排名,也就是第几小,现在我们要用pattern来匹配S.在本题 ...

随机推荐

  1. Selenium2+python自动化28-table定位

    前言 在web页面中经常会遇到table表格,特别是后台操作页面比较常见.本篇详细讲解table表格如何定位. 一.认识table 1.首先看下table长什么样,如下图,这种网状表格的都是table ...

  2. hdu_4897_Little Devil I(树链剖分)

    题目连接:http://acm.hdu.edu.cn/showproblem.php?pid=4897 题意:有三种操作,1是在树上的两个节点之间的路径改变当前的颜色,2是改变树上有且只有一个端点在u ...

  3. c# 执行js的方法

    http://www.cnblogs.com/wuhuacong/archive/2010/11/08/1871866.html 为了有效阻止恶意用户的攻击,一般登录都会采用验证码方式方式处理登录,类 ...

  4. jquery页面滑到底部加载更多

    $(window).scroll(function(){ var _scrolltop = $('body').scrollTop();if(_scrolltop+_winHeight>_doc ...

  5. POJ 1821 单调队列+dp

    题目大意:有K个工人,有n个墙,现在要给墙涂色.然后每个工人坐在Si上,他能刷的最大范围是Li,且必须是一个连续子区间,而且必须过Si,他刷完后能获得Pi钱 思路:定义dp[i][j]表示前i个人,涂 ...

  6. 11、组合模式(Composite)

    组合模式有时又叫部分-整体模式在处理类似树形结构的问题时比较方便,看看关系图: 直接来看代码: [java] view plaincopy public class TreeNode { privat ...

  7. 总结自己的Git常用命令

    总结自己的Git常用命令 使用git也有一段时间了,把自己常用的命令用自己的描述记录起来,方便自己备忘也方便其他人参考. 目录: 最基本的命令: git clone 拷贝并跟踪远程的master分支. ...

  8. H5 video跨域问题

    事情是这样的,今天上午下班前,朋友发我一个js文件,说视频不播放,核心代码大概是这样子的: var player = document.createElement('video'); var sour ...

  9. Android 之 ServiceManager与服务管理

    ServiceMananger是android中比较重要的一个进程,它是在init进程启动之后启动,从名字上就可以看出来它是用来管理系统中的service.比如:InputMethodService. ...

  10. CodeForces 616D Longest k-Good Segment

    用队列维护一下即可 #include<cstdio> #include<cstring> #include<queue> #include<algorithm ...