KMP算法讲解
老规矩,讲算法前,先说一道小问题吧
给你一个长串和短串,求短串在长串中出现的次数和位置。
设长串长度为len1,短串长度为len2。
如果len1*len2<=108,那就很简单了,直接暴力枚举以每个字符为开始的字符串是否匹配即可,复杂度为O(len1*len2);(是不是感觉太大了?)
如果将数据范围扩大到len1,len2<-106呢?
现在就开始介绍我们的KMP算法。
有了前面的问题,KMP要解决的是什么就自然出来了,KMP的复杂度达到的耸人听问的O(len1+len2)。
我们可以想想我们相对于暴力算法需要改进什么?
我们可以每一次失配(也就是匹配失败)的时候,不用每一次都从上一次的出发点只往后移动一个字符,可以跳啊!
我们可以预处理出每一次跳的位置来有利于节省复杂度啊。
这里我们就讲一讲怎么跳,以及怎么进行预处理。
1.怎么跳?
我们假设字符串为abaaba
我们如果在第二个a时失配了,我们应该怎么往前呢?
我们就可以可以把第一个a放在这一个位置继续匹配。
那么,如果是第四个a呢?
我们是不是就可以把第二个a放在这个位置呢?
大家可以看到,第最后一个字符到第三个a的字符串是aba,而第一个字符到第二个a的字符串是不是也是aba,它们不是一样的吗?
讲到这里,大家应该大概的明白了KMP是怎么跳的了吧。
我们记一个nxt数组,nxt[i]表示的是从第一个字符到第i个字符的最长前后缀的长度。看不懂没关系,举个例子。
假设字符串为abaaba
nxt[0]=0
nxt[1]=0(ab无前后缀)
nxt[2]=1(aba最长前后缀为a)
nxt[3]=1(abaa-----a)
nxt[4]=0(abaab--无)
nxt[5]=3(abaaba-aba)
2.初始化
问题来了,怎么用很少的时间复杂度来进行初始化呢?
很容易想到递推,怎么递推呢?
我们假设求出了前面的nxt,现在多了一个,我们就应该找一找了。
我们可以跳前一个位置的nxt,直到跳到一个位置后面有一个字符是所需要的,是那里的后面那一个字符。
每个这样递推就好了!
而查找的过程与初始化的过程类似,这里就不再赘述了。
下面上一份模板代码
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
char s1[],s2[];
int nxt[];
int main()
{
scanf("%s",s1);
scanf("%s",s2);
nxt[]=;
int len1=strlen(s1);
int len2=strlen(s2);
for(int i=,k=;i<len2;i++)
{
k=nxt[i-];
while(k>&&s2[k]!=s2[i]) k=nxt[k-];
if(s2[k]==s2[i]) k++;
nxt[i]=k;
}
for(int i=,j=;i<len1;i++)
{
while(j!=&&s1[i]!=s2[j]) j=nxt[j-];
if(s1[i]==s2[j]) j++;
if(j==len2)
{
printf("%d\n",i-j+);
}
}
for(int i=;i<len2;i++) printf("%d ",nxt[i]);
return ;
}
模板题:https://www.luogu.org/problemnew/show/3375
感谢大家的支持!
如果有不足之处,请尽管提出,本人不胜感激!
KMP算法讲解的更多相关文章
- 串的应用与kmp算法讲解--学习笔记
串的应用与kmp算法讲解 1. 写作目的 平时学习总结的学习笔记,方便自己理解加深印象.同时希望可以帮到正在学习这方面知识的同学,可以相互学习.新手上路请多关照,如果问题还请不吝赐教. 2. 串的逻辑 ...
- (转载)KMP算法讲解
网上找到了一篇详细讲解KMP字符串匹配算法,质量很高.特备忘于此. 摘自:http://blog.csdn.net/v_july_v/article/details/7041827 实现代码如下: / ...
- [转]KMP算法理解及java实现
这大概是我看的最好懂的KMP算法讲解了,不过我还只弄懂了大概思想,算法实现我到时候用java实现一遍 出处:知乎 https://www.zhihu.com/question/21923021/ans ...
- 【原创】通俗易懂的讲解KMP算法(字符串匹配算法)及代码实现
一.本文简介 本文的目的是简单明了的讲解KMP算法的思想及实现过程. 网上的文章的确有些杂乱,有的过浅,有的太深,希望本文对初学者是非常友好的. 其实KMP算法有一些改良版,这些是在理解KMP核心思想 ...
- BF、KMP、BM、Sunday算法讲解
BF.KMP.BM.Sunday算法讲解 字串的定位操作通常称作串的模式匹配,是各种串处理系统中最重要的操作之一. 事实上也就是从一个母串中查找一模板串,判定是否存在. 现给出四种匹配算法包括BF(即 ...
- POJ 3461 Oulipo[附KMP算法详细流程讲解]
E - Oulipo Time Limit:1000MS Memory Limit:65536KB 64bit IO Format:%I64d & %I64u Submit ...
- 字符串匹配KMP算法的讲解C++
转自http://blog.csdn.net/starstar1992/article/details/54913261 也可以参考http://blog.csdn.net/liu940204/art ...
- KMP算法入门讲解
字符串匹配问题.假设文本是一个长度为$n$的字符串$T$,模板是一个长度为$m$的字符串$P$,且$m\leq n$.需要求出模板在文本中的所有匹配点$i$,即满足$T[i]=P[0],T[I+1]= ...
- 串匹配算法讲解 -----BF、KMP算法
参考文章: http://www.matrix67.com/blog/archives/115 KMP算法详解 http://blog.csdn.net/yaochunnian/artic ...
随机推荐
- HDU 4911 Inversion 树状数组求逆序数对
显然每次交换都能降低1 所以求出逆序数对数,然后-=k就好了.. . _(:зゝ∠)_ #include<stdio.h> #include<string.h> #includ ...
- as 与 is
在存储过程(PROCEDURE)和函数(FUNCTION)中没有区别:在视图(VIEW)中只能用AS:在游标(CURSOR)中只能用IS.
- redis缓存的安装和配置
ubantu16.04环境下安装 下载安装,依次执行命令; # 从官方网站下载安装包,注意,当前在哪个目录下执行命令,下载的包将在哪个目录下 $ wget http://download.redis. ...
- Tomcat在Linux服务器上的BIO、NIO、APR模式设置
一.BIO.NIO.AIO 先了解四个概念: 同步 : 自己亲自出马持银行卡到银行取钱(使用同步IO时,Java自己处理IO读写). 异步 : 委托一小弟拿银行卡到银行取钱,然后给你(使用异步IO时, ...
- centos6.5安装git
1.git源码地址:http://codemonkey.org.uk/projects/git-snapshots/git/
- Intellij IDEA编译代码出现红色标志
如图 原因:项目没有模块化加载 解决方法: 快捷键 Ctrl+Alt+Shift+S 进入 Project Structure 界面,选择 Modules --> Sources -->选 ...
- git的使用,eclipse操作(待更新)
使用eclipse将项目上传到git私有服务器或github服务器: 1.配置个人信息: 2.配置eclipse私钥公钥: 上传公钥到仓库: 修改仓库中的公钥名称: 3.提交代码 一个仓库只能上传一个 ...
- 【java】对象变成垃圾被垃圾回收器gc收回前执行的操作:Object类的protected void finalize() throws Throwable
package 对象被回收前执行的操作; class A{ @Override protected void finalize() throws Throwable { System.out.prin ...
- UITableView的性能优化
UITableView作为ios中使用最频繁的控件之一,其性能优化也是常常要面对的,尤其是当数据量偏大并且设备性能不足时.本文旨在总结tableview的几个性能优化tips,并且随着认识的深入,本文 ...
- 首次在C#程序中用log4net
众所周知log4net是一个很强大的日志管理库,我自己也用了下,这里作下记录: 首先新建一个项目Log4NetTest,然后将log4net.dll程序集添加引用至Log4NetTest. 然后在Lo ...