Loading [MathJax]/jax/output/CommonHTML/config.js
前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >温故KMP算法

温故KMP算法

作者头像
ShenduCC
发布于 2018-04-27 04:18:13
发布于 2018-04-27 04:18:13
7030
举报
文章被收录于专栏:算法修养算法修养

最近由于某些原因,又回顾了一次KMP算法。上一次回顾KMP算法还是在刷题的时候遇到的:

http://blog.csdn.net/dacc123/article/details/50994611

在我的记忆力,每次回顾KMP算法都会有新的理解,以为自己理解的很透彻了,等过一段时间再去回顾,又要花一些时间去弄门清。这次也一样。

刚接触Next数组的时候我很反感字符串前缀和后缀的最长公共子串的长度来解释next数组,我认为next数组就是一个字符串的对称程度。在这样的理解之下,计算next数组的理解就是:

    在求解next数组的时候,若前面一个next数,为0,那么说明前面没有对称的,新加的字符如果要对称只可能和第一个字符开始比较。如果next数不为0,说明前面一个字符是有和它对称的,那么去找和他对称的字符的下一个字符,如果相等那么next值就++,如果不相等只能等于0了。

从今天看来,这个对称理解显然是错误的,很容把误导到回文串里面的前后对称。KMP算法其实很简单,就从前缀和后缀去理解他,这也是他算法的核心思想。

下面举个例子:

第一次匹配:从第0位开始,匹配到第7位都是相同的,最后一位发现不一样了就是第8位

0   1    2   3   4   5    6   7   8

a   b   c    x   y    a   b   c   x   y   a -------------目标字符串

a   b   c    x   y    a   b   c   1     -----------------模式字符串

接下来:

如果是暴力的话,应该是模式字符串向前移动一位,进行比较,发现第一位有不匹配的继续移动。

0   1    2   3   4   5    6   7   8

a   b   c    x   y    a   b   c   x   y  a -------------目标字符串

     a   b   c    x    y   a   b   c   1     -----------------模式字符串

假设暴力移动了x位,终于有可能匹配了,这里是有可能。那么情况一定是这样:

0   1    2   3   4   5    6   7   8

a   b   c    x   y    a   b   c   x   y   a -------------目标字符串

                         a   b   c   x   y   a   b   c   1     -----------------模式字符串

模式字符串的a , b ,c和目标的5,6,7位是相同的,(我们不看第8位以及后面的只看0~7)。这样才有可能匹配(前面移动的都是从第一位就pass掉了)。

那么回到第一步:

0   1    2   3   4   5    6   7   8

a   b   c    x   y    a   b   c   x   y   a -------------目标字符串

a   b   c    x   y    a   b   c   1     -----------------模式字符串

在发现第8位不匹配的时候,我们之前暴力推算过,向前移动5位,才有可能匹配。(只看0~7位)前7位都是相同的,我们可以找到规律,为什么移动5位才有可能匹配:

a   b   c   x   y   a   b  c

                       a   b  c   x    y   a   b  c

可以看这就是一个字符串的前缀=后缀的情况,不是吗?也就是说,只有当前缀等于后缀存在的情况下,你往后移才有可能匹配(在0~7之内有匹配的)。在发现第8位不匹配的情况下,我们利用next数组,直接找到前缀=后缀的那部分,直接移动过去,这样省了很多步暴力。如果发现前缀=后缀的情况不存在,那么好办,直接跳过0~7位,因为前缀=后缀不存在,你在0~7位之间怎么移动都不可能匹配。

接下来就是利用前缀与后缀求next数组的方法,很容易理解。

比如 s: a   b    a   b 

next[i]  表示的是从第0~i位的字符串,前缀和后缀的最大公共子串的长度。求解next[i] 其实只有两种情况,一种是next[i-1]也就是0~i-1的子串存在前后缀最大公共子串,例如a  b  a  b 现在求解最后一位b也就是next[3],可以看next[2]=1 因为a b a的公共前后缀是a长度是1,s[0]=s[2]="a" 。 那么如果s[1]=s[3]的话,公共前后缀岂不是要加1,于是b就去找s[2]匹配的前缀就是s[1],找他的下一位s[1],果然和自己相等,于是在next[2]的基础上加1.。还要一种就是前面的next[i-1]没有前后缀公共子串,那么看来只有从自己开始开辟了,忽视果断和第一位比较,如果相等,那么从i开始就有了前后缀公共子串,长度为1.

这里还要提一点,next[i] 还表示和s[i]相等的前缀s[j]的下标j,s[j]是前缀的最后一个字符,s[i]是后缀的最后一个字符。s[i]=s[j] ,j的值既是下标(从0开始的要加 1)也是长度。

next[0]   a   只有一个字符串,最大公共子串长度为0

next[1]   a b   由于next[0]=0,说明前面的子串没有前后缀相等的情况,只能从自己开辟,发现s[0]和自己不一样,于是只能next[1]=0

next[2]  a b a   next[1]=0,同样的从自己开辟,发现s[0]和自己一样,终于有戏,于是next[2]=1

next[3]  a b a b    next[2]=1 ,前面有匹配的,于是找到next[2]匹配的那个字符串下表也就是next[2]的值,是1(我这里是下标从0开始)于是找s[0]的下一位s[1]发现和自己一样,很完美,在next[2]的基础上加1。如果不一样呢,那么很认命,自己破坏了前后缀公共子串,只能是0.

至于代码什么的就不贴了,明白了原理,写代码是信手拈来的事情,对吧!

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2018-03-20 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
改进的模式匹配算法—KMP算法
在暴力匹配中,每趟匹配失败都是模式后移一位再从头开始比较。而某趟已匹配相等的字符序列是模式的某个前缀,这种频繁的重复比较相当于模式串在不断地进行自我比较,这就是低效率的根源。
一条晒干的咸鱼
2024/11/19
2220
改进的模式匹配算法—KMP算法
【算法】----BF算法&KMP算法
打开我们浏览器的搜索框,输入你想的这个词,然后点击Enter。浏览器就会自动搜索与该词匹配的内容。
Skrrapper
2024/06/18
1450
【算法】----BF算法&KMP算法
彻底搞懂KMP算法原理
也就是说,KMP算法是用来解决字符串匹配问题的,从一个主字符串text中寻找一个子字符串(模式字符串)pattern,看这个子串是否在主串中,比如对于text='abaacababcac'和pattern='ababc',子串是包含在主串中的,同时它在主串中的索引是5。
兜兜转转
2023/03/08
6.5K0
彻底搞懂KMP算法原理
数据结构篇——KMP算法
我们其中可以优化的点就是i的位置更新,我们可以根据p字符串的特性来判断i在失败后最近可以移动到哪个点位!
秋落雨微凉
2022/11/21
2590
KMP算法
Ⅱ.j=2,k=1,P[a]!=P[b],∴k = next[k]即k = next[1]=0,j=2不变
堆栈哲学
2022/11/24
8390
KMP算法
KMP算法的原理和实现
只要你学过数据结构与算法分析,相信你对KMP算法应该都不陌生吧?如果你没听过,不要紧,今天我们就来聊一聊这个算法。建议最好拿一张草稿纸,然后边看边理解,这样更有助于你对它的理解,更能理解它背后的精髓所在,相信你在理解完该算法之后,一定会大喊一声:妙啊!
xujjj
2020/05/18
6910
深入解析 Knuth-Morris-Pratt 算法:字符串匹配的高效解决方案
这篇文章主要是总结一下kmp算法。所以就不写暴力遍历的逻辑了。这个算法属实是让我看了挺长时间,各种讲解博客是一点也看不进去(不是写的不详细,而是总感觉写的乱七八糟很复杂),最长公共前缀一直没理解其作用,不过反反复复的刷对应的讲解视频,卒或有所闻。
f1sh
2024/07/23
2320
快速字符串匹配一: 看毛片算法(KMP)
由于需要做一个快速匹配敏感关键词的服务,为了提供一个高效,准确,低能耗的关键词匹配服务,我进行了漫长的探索。这里把过程记录成系列博客,供大家参考。
ShenduCC
2019/08/07
2.1K0
【C++】算法集锦(10)通俗讲kmp算法
第二轮,模式串向后挪动一位,和主串的第二个等长子串比较,发现第0位字符不一致:
看、未来
2021/09/18
5940
重学KMP!
https://leetcode-cn.com/problems/implement-strstr/
代码随想录
2021/07/16
5170
漫画:什么是KMP算法?
在字符串匹配算法的前两讲,我们分别介绍了暴力算法BF算法,利用哈希值进行比较的RK算法,以及尽量减少比较次数的BM算法,没看过的小伙伴可以点击下方链接:
用户1564362
2020/03/12
5080
【数据结构】您有一份KMP算法教学已到账,请注意查收!!!
在上一篇内容中,我们详细介绍了朴素模式匹配算法及其实现。朴素模式匹配算法简单的理解就是将主串中以每一个位序上的元素为开头的子串与模式串进行匹配,直到匹配成功,或者匹配完主串中的所有可能的子串。
蒙奇D索隆
2024/09/07
1240
【数据结构】您有一份KMP算法教学已到账,请注意查收!!!
串的模式匹配之KMP算法
在之前我们介绍过串的朴素模式匹配算法,基本思路就是用主串中的每一个子串和模式串匹配,若匹配失败,都是模式串后移一位再重新开始比较,将模式串序号j置为1。我们假设主串的长度为m,模式串的长度为n,那么在最坏的情况下,主串中每个子串都和模式串进行了匹配,时间复杂度就为O(mn)。
mumumu
2022/12/26
4090
串的模式匹配之KMP算法
KMP算法(字符串匹配问题)
注意,是KMP算法,不是MMP哈,我没有骂人。KMP算法是用来做字符串匹配的,除了KMP算法分,还有暴力匹配算法,也是用来做字符串匹配的。接下来先看看暴力匹配算法,你就知道为啥会出现KMP算法了。
贪挽懒月
2021/02/02
4360
超详细!从本质上搞懂困惑你多年的KMP匹配算法
KMP算法是一种字符串匹配算法,可以在 O(n+m) 的时间复杂度内实现两个字符串的匹配。本文将引导您学习KMP算法。
帅地
2020/03/04
1K0
超详细!从本质上搞懂困惑你多年的KMP匹配算法
KMP算法-之next数组-详解
我们在一个母字符串中查找一个子字符串有很多方法。KMP是一种最常见的改进算法,它可以在匹配过程中失配的情况下,有效地多往后面跳几个字符,加快匹配速度。
MickyInvQ
2020/09/27
7.4K2
KMP算法-之next数组-详解
KMP算法复习
太久没打了,刚好有道题用上了,就复习一下。 我觉得复到KMP应该就够用了,如果要AC自动机我直接死在那里。
千灵域
2022/06/17
1810
KMP算法
下面就是重复上面的重复行为,如果对当前i和j位置进行比较发生了失配,那么i不变,j去找到在next数组中对应当前位置的值,然后把j移动到该位置,再把j当前指向位置与i指向位置进行比较
大忽悠爱学习
2021/03/15
4850
对KMP算法中next数组的深入理解(这个算法真有点难懂)
首先了解kmp算法是干嘛的,它的作用是进行一个模式匹配,即在一个字符串中寻找是否存在某一个子串,比如在aabbccabc这个主串中是否存在abc这个模式串,并且输入他们匹配时,在主串的位置,如上例中,就应该输出的是“在第7个位置他们进行匹配”。 这就是kmp算法的作用。
戈贝尔光和热
2018/12/27
4.2K0
一文理解kmp算法(java代码)
对于模式串中已经匹配过的那些字符,如果我们能找到一些规律,将模式串多往后移动几位,而不是像暴力算法算法一样,每次把模式串移动一位,就可以提高算法的效率。kmp算法给我们提供的思路是:对于模式串,将每一个字符在匹配失败时可以向后移动的最大距离保存在一个next数组。这样当匹配失败时就可以按照next数组中保存的数字向后多移动几位。从而提高算法的效率。
用户10604450
2024/03/15
1950
一文理解kmp算法(java代码)
相关推荐
改进的模式匹配算法—KMP算法
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档