(注:该贴主要运用python实现该算法)

  先谈谈KMP算法吧。KMP算法的全称是Knuth-Morris-Pratt 算法,它是用来进行字符串查找,即在某个主字符串里面找到某个特定子字符串。但是好像这个问题也可以直接暴力查找来完成啊,可是暴力查找的的缺点是不可忽视的:它的时间复杂度太高了!一旦遇见长的字符串就会让程序运行时间指数型增长。而用KMP算法可以很好的解决代码的时间复杂度高的问题,它的时间复杂度是线性的,也就是说该算法的时间复杂度取决于两个字符串的长度。

  接下来我会对KMP算法完成任务的大概思路进行叙述

  首先,我们约定一些符号:S为主字符串,也就是被进行查找的字符串;P为子字符串,也就是需要查找的字符串;next为next数组,里面记录了一些解决任务的关键信息,这里先买一些关子,毕竟比较难解释。

  然后就是给定一个主字符串S = ‘ACBACC DBACBACDEA’,子字符串P = ‘ACBACD’,next = [-1, 0, 0, 0, 1, 2]

  接着开始比对

  如上图,当i = 0,j = 0时,二者相等,所以i和j皆进一位;

       当i = 1,j = 1时,二者相等,所以i和j皆进一位;

       当i = 2,j = 2时,二者相等,所以i和j皆进一位;

       当i = 3,j = 3时,二者相等,所以i和j皆进一位;

       当i = 4,j = 4时,二者相等,所以i和j皆进一位;

       当i = 5,j = 5时,二者不相等,所以把j = next[j] = 3,i不变;

(箭头表示当前在比较的位置)

       当i = 5,j = 2时,二者相等,所以i和j皆进一位;

       当i = 6,j = 3时,二者不相等,所以把j = next[j] = 0,i不变;

(箭头表示当前在比较的位置)

       当i = 6,j = 0时,二者不相等,所以把j = next[j] = -1,i不变;

       当i = 6,j = -1时,此时j为特殊值,所以i和j皆进一位;

       当i = 7,j = 0时,二者不相等,所以把j = next[j] = -1,i不变;

       当i = 7,j = -1时,此时j为特殊值,所以i和j皆进一位;

       当i = 8,j = 0时,二者不相等,所以把j = next[j] = -1,i不变;

       当i = 8,j = -1时,此时j为特殊值,所以i和j皆进一位;

(箭头表示当前在比较的位置)

      当i = 9,j = 0时,二者相等,所以i和j皆进一位;

      当i = 10,j = 1时,二者相等,所以i和j皆进一位;

       当i = 11,j = 2时,二者相等,所以i和j皆进一位;

      当i = 12,j = 3时,二者相等,所以i和j皆进一位;

      当i = 13,j = 4时,二者相等,所以i和j皆进一位;

       当i = 14,j = 5时,二者相等,所以i和j皆进一位;

      当i = 15,j = 6时,此时检测到j>len(P)了,则跳出循环;

       最后返回布尔值,或者返回你想要得到的信息

  如此,我们就走完了一次KMP算法,完成了一次任务,得到了正确的结果

  

  通过上面的流程,我们可以得知KMP算法中有一个重要的部分:next数组。

  那next数组是什么呢?next数组主要用于存储j位之前的字符串的最长相同前缀和后缀的长度。

  什么是前缀、后缀呢?"前缀"指除了最后一个字符以外,一个字符串的全部头部组合;"后缀"指除了第一个字符以外,一个字符串的全部尾部组合。当然,这里指的是在j位之前包括j位的前后缀。

  需要注意的是:假如有一个字符串“abcd”,那么其前缀是:a ab abc,其后缀是:bcd cd d。也就是说前后缀是不止一个的。

  而前文所说的最长相同前缀和后缀的长度即是指:假若有一个字符串“aabab”,其前缀是:a aa aab aaba,其后缀是:aaba aba ba a,那这个的最长相同前后缀是a,所以该位置对应next数组的位置的值的应该是1。

  练习:“abcabx”  [0,0,0,1,2,0]

 )

  这里提供一个代码计算next数组的方法

def get_next(son_str: str) -> list():
"""
获得next数组 参数解释 son_str: 需要求next数组的字符串
返回值: 返回next数组
"""
length = len(son_str) # 定义next数组
next = length*[None]
next[0] = -1
next[1] = 0 # 计算next数组
k = -1
j = 0
while j < length-1:
if son_str[k] == son_str[j] or k == -1:
j += 1
k += 1
next[j] = k
else:
k = next[k]
return next

  这里的next[0] = -1主要是因为方便代码处理j回到0时,发现S[i] != P[j]时,i无法进位的情况(用上面第一个方法求出的next数组也可用,但是具体方法得去搜索了,作者是使用的是代码求出来的那个next数组)

  到此,该算法也已经讲得差不多了

  下面提供完整的代码

#!/usr/bin/env python
# -*- encoding: utf-8 -*-
'''
@文件名 : KMP.py
@描述 : 实现KMP算法,进行字符串比对
@创建时间 : 2023/09/07/20
@作者 : zrold
@版本 : 1.0
''' def kmp(farther_str: str, son_str: str) -> bool:
"""
定义KMP算法, 并根据传进来的两个参数来进行比对, 并返回一个布尔值 参数解释: farther_str: 进行比对的主字符串,
son_str: 子字符串
返回值: 返回一个布尔值
"""
# 得到next数组
next = get_next(son_str) # 匹配字符串
i = 0
j = 0
while i < len(farther_str) and j < len(son_str):
if farther_str[i] == son_str[j] or j == -1:
i += 1
j += 1
else:
j = next[j] if j >= len(son_str):
return True
else:
return False def get_next(son_str: str) -> list():
"""
获得next数组 参数解释 son_str: 需要求next数组的字符串
返回值: 返回next数组
"""
length = len(son_str) # 定义next数组
next = length*[None]
next[0] = -1
next[1] = 0 # 计算next数组
k = -1
j = 0
while j < length-1:
if son_str[k] == son_str[j] or k == -1:
j += 1
k += 1
next[j] = k
else:
k = next[k]
return next if __name__ == '__main__':
farther_str = input('请输入需要进行对比的主字符串:')
son_str = input('请输入需要在主字符串中找到的子字符串:')
if kmp(farther_str, son_str):
print(f'确实存在"{son_str}"在"{farther_str}"中')
else:
print(f'不存在"{son_str}"在"{farther_str}"中')

  

KMP字符串对比算法及next数组计算的更多相关文章

  1. 字符串(马拉车算法,后缀数组,稀疏表):BZOJ 3676 [Apio2014]回文串

    Description 考虑一个只包含小写拉丁字母的字符串s.我们定义s的一个子串t的“出 现值”为t在s中的出现次数乘以t的长度.请你求出s的所有回文子串中的最 大出现值. Input 输入只有一行 ...

  2. KMP字符串查找算法

    #include <iostream> #include <windows.h> using namespace std; void get_next(char *str,in ...

  3. 字符串类——KMP子串查找算法

    1, 如何在目标字符串 s 中,查找是否存在子串 p(本文代码已集成到字符串类——字符串类的创建(上)中,这里讲述KMP实现原理) ? 1,朴素算法: 2,朴素解法的问题: 1,问题:有时候右移一位是 ...

  4. poj 2406:Power Strings(KMP算法,next[]数组的理解)

    Power Strings Time Limit: 3000MS   Memory Limit: 65536K Total Submissions: 30069   Accepted: 12553 D ...

  5. 字符串匹配KMP算法中Next[]数组和Nextval[]数组求法

    数据结构课本上给了这么一段算法求nextval9[]数组 int get_nextval(SString T,int &nextval[ ]) { //求模式串T的next函数修正值并存入数组 ...

  6. KMP算法的next[]数组通俗解释

    原文:https://blog.csdn.net/yearn520/article/details/6729426 我们在一个母字符串中查找一个子字符串有很多方法.KMP是一种最常见的改进算法,它可以 ...

  7. hdu 1358:Period(KMP算法,next[]数组的使用)

    Period Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Subm ...

  8. KMP 算法 & 字符串查找算法

    KMP算法 Knuth–Morris–Pratt algorithm 克努斯-莫里斯-普拉特 算法 algorithm kmp_search: input: an array of character ...

  9. KMP字符串模式匹配详解(zz)

    刚看到位兄弟也贴了份KMP算法说明,但本人觉得说的不是很详细,当初我在看这个算法的时候也看的头晕昏昏的,我贴的这份也是网上找的.且听详细分解: KMP字符串模式匹配详解 来自CSDN     A_B_ ...

  10. KMP字符串模式匹配详解(转)

    来自CSDN     A_B_C_ABC 网友 KMP字符串模式匹配通俗点说就是一种在一个字符串中定位另一个串的高效算法.简单匹配算法的时间复杂度为O(m*n);KMP匹配算法.可以证明它的时间复杂度 ...

随机推荐

  1. Multiserver游戏服务器Demo[C++&Lua]

    代码参考 代码文件参考下述详解的类图,工程参考第零章工程说明 关键特性 对Socket库进行封装,抹平Socket的Window&Linux的平台差异. C++嵌入lua脚本,增加开发者编码效 ...

  2. htop 和 bashtop 的一些不足

    htop 和 bashtop 都是 Linux 资源监视器中非常好用的工具,尤其对于展示当前 Linux 操作系统的处理器.内存.硬盘.网络和进程等各项资源的使用情况与状态.但它们都有一个问题,就是当 ...

  3. 为什么 Biopython 的在线 BLAST 这么慢?

    用过网页版本 BLAST 的童鞋都会发现,提交的序列比对往往在几分钟,甚至几十秒就可以得到比对的结果:而通过调用 API 却要花费几十分钟或者更长的时间!这到底是为什么呢? NCBIWWW 基本用法 ...

  4. Windows 10 开启子系统Ubuntu

    卸载原有的wsl 分发子系统 # 查看已安装的wsl子系统 wsl --list # 依次删除wsl 子系统 wsl --unregister <子系统名称> 结果 安装子系统Ubuntu ...

  5. C#/.Net的多播委托到底是啥?彻底剖析下

    前言 委托在.Net里面被托管代码封装了之后,看起来似乎有些复杂.但是实际上委托即是函数指针,而多播委托,即是函数指针链.本篇来只涉及底层的逻辑,慎入. 概括 1.示例代码 public delega ...

  6. MVCC并发版本控制之重点ReadView

    MVCC并发版本控制 本文大部分来自<MySQL是怎样运行的>,这里只是简单总结,用于各位回忆和复习. 版本链 对于使用 InnoDB 存储引擎的表来说,它的聚簇索引记录中都包含两个必要的 ...

  7. tSNE算法在自然语言处理中的应用:文本降维和可视化

    目录 技术原理及概念 t-SNE(Toeplitz-Stochastic Neural Network)是一种常用的文本降维和可视化算法,它的核心思想是将高维文本数据映射到低维空间,同时保持数据的一致 ...

  8. Paimon Compaction实现

    Compact主要涉及以下几个组件 CompactManager 管理Compact task CompactRewriter 用于compact过程中数据的重写实现, 比如compact过程中产生c ...

  9. Serverless试飞员的夙愿 | 带您扶摇直上,酣畅淋漓的云上作战

    ​上期博文带您体验了外挂云函数Demo包,感受通过云函数使用云数据库快速突破"音障",进入"长机"云函数+"僚机"云数据库的Serverle ...

  10. 使用 JCommander 解析命令行参数

    前言 如果你想构建一个支持命令行参数的程序,那么 jcommander 非常适合你,jcommander 是一个只有几十 kb 的 Java 命令行参数解析工具,可以通过注解的方式快速实现命令行参数解 ...