飘逸的python - 字符串的KMP匹配算法

来源:互联网 发布:steam淘宝买游戏怎么搞 编辑:程序博客网 时间:2024/04/29 17:11
首先我们来看一下字符串的朴素匹配.

可以想象成把文本串s固定住,模式串p从s最左边开始对齐,如果对齐的部分完全一样,则匹配成功,失败则将模式串p整体往右移1位,继续检查对齐部分,如此反复.

#朴素匹配def naive_match(s, p):    m = len(s); n = len(p)    for i in range(m-n+1):#起始指针i        if s[i:i+n] == p:            return True    return False

关于kmp算法,讲的最好的当属阮一峰的<字符串匹配的KMP算法>.一路读下来,豁然开朗.
其实就是,对模式串p进行预处理,得到前后缀的部分匹配表,使得我们可以借助已知信息,算出可以右移多少位.即 kmp = 朴素匹配 + 移动多位.
更多细节请看阮一峰的文章,这里就不展开了.
下面给出python的代码实现.

#KMPdef kmp_match(s, p):    m = len(s); n = len(p)    cur = 0#起始指针cur    table = partial_table(p)    while cur<=m-n:        for i in range(n):            if s[i+cur]!=p[i]:                cur += max(i - table[i-1], 1)#有了部分匹配表,我们不只是单纯的1位1位往右移,可以一次移动多位                break        else:            return True    return False#部分匹配表def partial_table(p):    '''partial_table("ABCDABD") -> [0, 0, 0, 0, 1, 2, 0]'''    prefix = set()    postfix = set()    ret = [0]    for i in range(1,len(p)):        prefix.add(p[:i])        postfix = {p[j:i+1] for j in range(1,i+1)}        ret.append(len((prefix&postfix or {''}).pop()))    return retprint naive_match("BBC ABCDAB ABCDABCDABDE", "ABCDABD")print partial_table("ABCDABD")print kmp_match("BBC ABCDAB ABCDABCDABDE", "ABCDABD")


3 0
原创粉丝点击
热门问题 老师的惩罚 人脸识别 我在镇武司摸鱼那些年 重生之率土为王 我在大康的咸鱼生活 盘龙之生命进化 天生仙种 凡人之先天五行 春回大明朝 姑娘不必设防,我是瞎子 二审上诉被驳回怎么办 醉酒驾车取保候审以后怎么办 小案子证据不足怎么办 撞车不严重逃逸怎么办 被执行人没有财产执行怎么办 挖到人头了怎么办 取保保证金不退怎么办 被诬陷经济诈骗怎么办 醉驾刑事拘留后怎么办 被别人举报赌博怎么办 涉黄刑事拘留怎么办取保候审 换了车牌保险怎么办 车辆转让后保险怎么办 立案后警察不管怎么办 打架后对方讹人怎么办 工商被恶意举报怎么办 店铺被工商举报怎么办 被买单侠恐吓怎么办? 团伙作案刑事拘留怎么办取保 欢乐麻将老输怎么办 回不了家怎么办身份证 没注意闯红灯了怎么办 摩托车被套牌了怎么办? 发现员工偷钱怎么办 盗窃刑事拘留7天怎么办 盗窃抓不到人怎么办 发现宿舍被盗后怎么办(  ) 发现宿舍被盗后怎么办() 回收到赃物电瓶怎么办 不知情买了赃物怎么办 盗窃单位要报警怎么办 上网吸烟被逮住怎么办 有人在微信骂我怎么办 网吧抽烟被拍照怎么办 诈骗被拘留该怎么办 行政拘留人跑了怎么办 车牌号被偷了怎么办 当员工提出辞职怎么办 老员工提出辞职怎么办 家人进看守所了怎么办 被贷款中介诈骗怎么办