kmp算法简明教程
在字符串s中寻找模式串p的位置,这是一个字符串匹配问题。
举例说明:
i = 0 1 2 3 4 5 6 7 8 9 10 11 12 13
s = a b a a c a b a a a b a a b
p = a b a a b
j = 0 1 2 3 4
在kmp算法发明之前。人们使用这种算法:
'''
原始的求子串位置算法,O( m * n )
'''
def string_index_of( pstr, pattern, pos = 0 ):
str_index = pos
pattern_index = 0
pattern_len = len( pattern )
while str_index < len( pstr ) and pattern_index < pattern_len:
if pstr[ str_index ] == pattern[ pattern_index ]:
str_index += 1
pattern_index += 1
else:
str_index = str_index - pattern_index + 1
pattern_index = 0
if pattern_index == pattern_len:
return str_index - pattern_index
return -1 pstr = 'i am caochao, i love coding!'
pattern = 'ao'
print( string_index_of( pstr, pattern, 7 ) )
print( pstr.find( pattern ) )
当s[4]与p[4]失配时,主串s回溯到i=1,模式串回溯到j=0,然后从此位置继续匹配。显然这种做法效率低下,假设s长度为n,p长度为m,则其时间复杂度为O(m*n)。
现考虑这样一个问题,当s与p匹配到位置i,j处,s[j]不等于p[j],如果此时保持i不变,p串中从k(0<k<j)处继续匹配,这样无需回溯i的做法这就是我们要讲到的kmp算法。那么应当如何取得这个k值?可以预先求出p中每个失配的j处需要跳转到的位置k(next[j]),这就是kmp算法中的next数组。
kmp算法步骤如下:
1,初始化i,j均为0,
2,依次往后比较s[i]与p[j],若相等则i,j各自加1,否则保持i不变,j=k(next[j])。若某时刻求得j值为-1,i,j也各自加1然后继续匹配
3,重复步骤2
依据上述分析可知,kmp算法中最关键之处在于k值的取法。在匹配进行到s[i]不等于p[j]时,假设应当让s[i]与p[k]继续比较(0<k<j),那么p中前k个字符必须满足,且不能存在k'>k满足等式1:
等式1,p[0,k-1]=s[i-k,i-1]
而在i,j之前已经匹配的字符里存在等式2:
等式2,p[j-k,j-1]=s[i-k,i-1]
由此,可以推出等式3:
等式3,p[0,k-1]=p[j-k,j-1]
至此,k值的取法已经非常明显,即在p串中取最大的k(0<k<j),使得p中开始的k个字符与p[j]之前的k个字符相等。这样就可在s[i]不等于p[j]时,尽可能在距离p[0]远的位置处继续匹配,从而提高匹配效率。
从上面的分析中可以给出k,即next[j]的定义:
1,j=0时,next[j]=-1
2,next[j] = max{k|0<k<j且p[0,k-1]=p[j-k,j-1]}
3,其它情况,next[j]=1
递推求next数组:
从next[j]的定义出发,可以采用递推的方式求得next[j]:
首先,next[0]=-1
令next[j]=k(0<k<j),则表明在p中存在k,且不存在k'>k满足下列关系:
p[0,k-1]=p[j-k,j-1]
那么next[j+1]的取值有3种情况,
1,若p[k]=p[j],则表明在p中存k,且不存在k'>k满足关系p[0,k]=p[j-k,j],那么next[j+1]=k+1,即
next[j+1]=next[j]+1
2,若p[k]不等于p[j],此时可把求next函数的过程看成模式匹配的过程,即p既是主串又是模式串。而在模式匹配过种中,此时应当让p[j]与p[next[k]]继续比较。
为了便于理解,这里令next[k]=k'。
若p[j]=p[k']时,next[j+1]=k'+1,即next[j+1]=next[k]+1,也即
next[j+1]=next[next[j]]+1
同理若p[j]不等于p[k'],那么继续让p[j]与next[k']比较,依次类推,直至k'=-1时:
next[j+1]=0
代码实现如下:
'''
kmp求next[j]数组
'''
def kmp_get_next( pattern ):
i = 0
j = -1
_next = [ 0 ] * len( pattern )
_next[ 0 ] = -1
while i < len( pattern ) - 1:
if j == -1 or pattern[ i ] == pattern[ j ]:
i += 1
j += 1
_next[ i ] = j
else:
j = _next[ j ]
return _next
优化的求next数组方法:
考虑如下模式串:
j = 0 1 2 3 4
p = a a a a b
next[j] = -1 0 1 2 3
若某时刻s[i]与p[3]不相等,依据next[3]指示应当让s[i]与p[2]继续比较,因p[3]与p[2]相等,这一步明显是多余的。推广到普遍情况,在求next数组过程中,如果next[i]=j且p[i]=p[j],那么令next[i]=next[j]。代码如下:
'''
kmp求next[j]数组
'''
def kmp_get_next( pattern ):
i = 0
j = -1
_next = [ 0 ] * len( pattern )
_next[ 0 ] = -1
while i < len( pattern ) - 1:
if j == -1 or pattern[ i ] == pattern[ j ]:
i += 1
j += 1
if pattern[ i ] == pattern[ j ]:
_next[ i ] = _next[ j ]
else:
_next[ i ] = j
else:
j = _next[ j ]
return _next
预先求得next[j]数组后,kmp算法代码实现如下:
'''
kmp求子串位置
'''
def kmp_index_of( pstr, pattern, pos = 0 ):
_next = kmp_get_next( pattern )
str_index = pos
pattern_index = 0
pattern_len = len( pattern )
while str_index < len( pstr ) and pattern_index < pattern_len:
if pattern_index == -1 or pstr[ str_index ] == pattern[ pattern_index ]:
str_index += 1
pattern_index += 1
else:
pattern_index = _next[ pattern_index ]
if pattern_index == pattern_len:
return str_index - pattern_index
return -1 pstr = 'i am caochao, i love coding!'
pattern = 'ao'
print( kmp_index_of( pstr, pattern, 7 ) )
print( pstr.find( pattern ) )
最后,对比下普通算法与kmp算法解决本文开始提出的问题匹配过程:
普通算法:
i = 0 1 2 3 4 5 6 7 8 9 10 11 12 13
s = a b a a c a b a a a b a a b
p = a b a a b
j = 0 1 2 3 4
s[4]不等于p[4],令i=1,j=0
i = 0 1 2 3 4 5 6 7 8 9 10 11 12 13
s = a b a a c a b a a a b a a b
p = a b a a b
j = 0 1 2 3 4
s[1]不等于p[0],令i=2,j=0
i = 0 1 2 3 4 5 6 7 8 9 10 11 12 13
s = a b a a c a b a a a b a a b
p = a b a a b
j = 0 1 2 3 4
s[3]不等于p[1],令i=3,j=0
i = 0 1 2 3 4 5 6 7 8 9 10 11 12 13
s = a b a a c a b a a a b a a b
p = a b a a b
j = 0 1 2 3 4
限于篇幅,略过中间n步,跳至i=9,j=0
i = 0 1 2 3 4 5 6 7 8 9 10 11 12 13
s = a b a a c a b a a a b a a b
p = a b a a b
j = 0 1 2 3 4
一路i++,j++,直到i=14,跳出循环结束匹配,并返回9。
kmp算法:
p串next数组为:
j = 0 1 2 3 4
p = a b a a b
next[j] = -1 0 0 1 1
next数组优化过后变为:
j = 0 1 2 3 4
p = a b a a b
next[j] = -1 0 -1 1 0
下面开始匹配:
i = 0 1 2 3 4 5 6 7 8 9 10 11 12 13
s = a b a a c a b a a a b a a b
p = a b a a b
j = 0 1 2 3 4
s[4]不等于p[4],令j=0
i = 0 1 2 3 4 5 6 7 8 9 10 11 12 13
s = a b a a c a b a a a b a a b
p = a b a a b
j = 0 1 2 3 4
s[4]不等于p[0],next[0]=-1,因此i,j各自加1。i=5,j=0
i = 0 1 2 3 4 5 6 7 8 9 10 11 12 13
s = a b a a c a b a a a b a a b
p = a b a a b
j = 0 1 2 3 4
i++,j++,直到s[9]不等于p[4],令j=0
i = 0 1 2 3 4 5 6 7 8 9 10 11 12 13
s = a b a a c a b a a a b a a b
p = a b a a b
j = 0 1 2 3 4
一路i++,j++,直到i=14,跳出循环结束匹配,并返回9。
通过对比可以看出,kmp算法比普通算法快得多,只要预先求出模式串next数组,则整个匹配过程中i无需回溯,时间复杂度亦由普通算法O(m*n)提升为O(m+n)。
kmp算法简明教程的更多相关文章
- 【★】KMP算法完整教程
KMP算法完整教程 全称: Knuth_Morris_Pratt Algorithm(KMP算法) 类型: ...
- 【★】KMP算法完整教程
KMP算法完整教程 全称: Knuth_Morris_Pratt Algorithm(KMP算法) 类型: ...
- KMP算法完整教程 (上)
KMP算法完整教程 全称: Knuth_Morris_Pratt Algorithm(KMP算法) 类型: 高级检索算法 功能: 字符串匹配查找 提出者: D.E.Knuth(克努兹),J.H.Mor ...
- !KMP算法完整教程
KMP算法完整教程 全称: Knuth_Morris_Pratt Algorithm(KMP算法) 类型: ...
- KMP算法完整教程 (下)
下面我们用数学归纳法来解决这个填值的问题. 这里我们借鉴数学归纳法的三个步骤(或者说是动态规划?): 1.初始状态 2.假设第j位以及第j位之前的我们都填完了 3.推论第j+1位该怎么填 初始状态我们 ...
- KMP算法简明法则
KMP算法也算是相当经典,但是对于初学者来说确实有点绕,大学时候弄明白过后来几年不看又忘记了,然后再弄明白过了两年又忘记了,好在之前理解到了关键点,看了一遍马上又能理解上来.关于这个算法的详解网上文章 ...
- Lisp简明教程
此教程是我花了一点时间和功夫整理出来的,希望能够帮到喜欢Lisp(Common Lisp)的朋友们.本人排版很烂还望多多海涵! <Lisp简明教程>PDF格式下载 <Lisp简明教程 ...
- KMP算法(研究总结,字符串)
KMP算法(研究总结,字符串) 前段时间学习KMP算法,感觉有些复杂,不过好歹是弄懂啦,简单地记录一下,方便以后自己回忆. 引入 首先我们来看一个例子,现在有两个字符串A和B,问你在A中是否有B,有几 ...
- Vbs 脚本编程简明教程之一
—为什么要使用 Vbs ? 在 Windows 中,学习计算机操作也许很简单,但是很多计算机工作是重复性劳动,例如你每周也许需要对一些计算机文件进行复制.粘贴.改名.删除,也许你每天启动 计算机第一件 ...
随机推荐
- Java多线程之 ThreadLocal
一.什么是ThreadLocal? 顾名思义它是local variable(线程局部变量).它的功用非常简单,就是为每一个使用该变量的线程都提供一个变量值的副本,是每一个线程都可以独立地改变自己的副 ...
- httpd.conf 禁止运行PHP和html页面
<VirtualHost *:80> ServerName www.test.com DocumentRoot /var/www/www.test.com ErrorDo ...
- 【Excel】Retrieving the COM class factory for component with CLSID {00024500-0000-0000-C000-000000000046}:
[Excel]Retrieving the COM class factory for component with CLSID {00024500-0000-0000-C000-0000000000 ...
- UVALive 7455 Linear Ecosystem (高斯消元)
Linear Ecosystem 题目链接: http://acm.hust.edu.cn/vjudge/contest/127401#problem/B Description http://7xj ...
- GRUB加密
在 /etc/grub.conf 内添加password=密码(也可使用加密的密码password= --md5 加密过的密码) 如何获得加密密码? 那就是grub-md5-crypt命令 简单流程如 ...
- .net 配置ueditor
添加引用如下: <script src="../Ueditor/ueditor.config.js" type="text/javascript"> ...
- Uvalive 4865 Data Recovery 最大流
题意就是 给一个50 * 50的矩阵 然后给出每行每列元素的和 和一个初始矩阵 矩阵中有些是未知,有些是已知 然后我们求目标矩阵就是把能确定的元素的值求出来,实在不能确定的就置为-1 所有矩阵元素的值 ...
- 找不到类型或命名空间 datarowview
在绑定数据时经常会用到这个句程序:<%# DataBinder.Eval(Container.DataItem,"xxxx")%>或者<%# DataBinder ...
- android 绘图之Canvas,Paint类
Canvas,Paint 1.在android 绘图但中经常要用到Canvas和Paint类,Canvas好比是一张画布,上面已经有你想绘制图画的轮廓了,而Paint就好比是画笔,就要给Canvas进 ...
- mac 开发必备软件(不断update ing...)
整理下mac环境下, 开发必备的一些软件吧, 由于不断要更新ing, 用到啥就写啥~球轻拍 1.host 绑定切换神器 a.gas mask : 只能切换单个自定义的host文件 b.ihosts(推 ...