【数据结构】KMP算法(详解)
创始人
2025-05-30 04:17:38
0

目录

  • 1. 朴素的模式匹配
  • 2. KMP算法解决的问题
  • 3. KMP算法
    • 公共前后缀(重点)
    • next 数组
    • KMP算法实现

1. 朴素的模式匹配

  • 朴素算法中,当匹配到不同位时,主串指针i会退回到该次匹配起点处的下一位置,以其为下一次匹配的主串起点

  • 同时字串的j指针退回其起始位置

  • 如此一来每次匹配主串指针后移一位,字串指针始终在其起始位置

  • 时间复杂度为O(m*n)

在这里插入图片描述

2. KMP算法解决的问题

  • 可以发现下图中,在第二次匹配时,第一个元素就已经不一样了

  • 朴素算法的缺点就在于其会傻傻的执行许多次这样不必要的判断

  • 这就是KMP算法所解决的问题

在这里插入图片描述

3. KMP算法

  • 主串指针不会进行回溯,不会回到朴素匹配中的下一匹配点
  • 利用已匹配部分中的公共前后缀来调整字串指针位置,以此加速下一次匹配

根据下面的动画感受感受
在这里插入图片描述

  • 可以看到,主串指针( i )在整个查找过程中都没有前移,每次查找的起点均为上次查找的结束点,即 i 永远不递减,这也使KMP的精髓
  • 同时,当不匹配位置前一位对应的next数组中元素不为0时,字串指针( j )会向后偏移相应个数的字符
  • 这样一来,无论是主串还是字串的判断次数都得到了优化,时间复杂度优化至O(m+n)

公共前后缀(重点)

公共前后缀的计算:
这里用公式理解,计算下标为a处的公共前后缀个数,如果[a-x,a]范围的每一个元素与[0,x]范围的每一个元素相等,则a处的公共前后缀个数为x+1

在这里插入图片描述
 

这里注意找某一位置的公共前后缀时,要将起始位置的字符同该位置字符比较,而不是只要在该位置之前出现了相同元素就判断存在公共前后缀
如下图中的红色位置B,虽然在其之前存在一个字符B,但是该位置的公共前后缀为0

在这里插入图片描述

 
 
 

next 数组

理解了什么是公共前后缀,其实next数组就是存储该数组每个对应位置公共前后缀数量的数组
 

(这里的next数组实际上为PM表,PM表右移一位 (空缺的用-1填充,最后一个元素的部分匹配值用于下一个元素,但没有下一个元素故可以舍弃) 并加一得到next数组。)
在这里插入图片描述
在这里插入图片描述
next表的含义是子串的第j个字符发生失配时跳到子串的next[j]位置重新与主串当前位置进行比较。

 

代码实现next数组(PM表)
这里其实用到了双指针,i 为next数组的下标,j 指向字串中的元素

int* get_next(const char* p)
{assert(p);int len = strlen(p);int* next = (int*)malloc(sizeof(int) * len);if (next == NULL)//满足vs2019及以上动态开辟检查标准,不写会报错{printf("malloc fail\n");exit(-1);//申请失败,异常退出}else{//先将其全部初始化为0memset(next, 0, sizeof(int) * len);int j = 0;//字串字符指针int i = 0;//相应位置next指针for (i = 1; i < 6; i++){if (p[i] == p[j]){next[i] = next[i - 1] + 1;//连续相等则next值递增j++;//字串字符指针依次向后偏移}else{//不相等则跳回字串起点,再判断是否与首位相同j = 0;if (p[i] == p[j]){next[i] =  1;//相同直接赋 1j++;}}}return next;}
}

 
 

KMP算法实现

注意代码注释

int my_kmp(char* a1, char* a2)
{int* next = get_next(a2);int len1 = strlen(a1);int len2 = strlen(a2);int i = 0;int j = 0;while (i < len1){if (a1[i] == a2[j]){//相等则两指针均向后偏移i++;j++;}else if (j > 0)  // j>0 时,根据next数组调整 j 的位置j = next[j - 1];else   //字串第一个字符就不匹配i++;if (j == len2) //匹配成功,返回值为字串第一个字符在主串中的下标return i - j;}return -1;
}

相关内容

热门资讯

“父母修养”之七:《争取“成功... 据报道,每年高考过后,考生家长离婚率猛增。据统计,2009年、2010年和2011年,每年高考结束后...
教书的是老师,但育人的一定是父... 俞敏洪《教书的是老师,但育人的一定是父母》的文章在网络上广为传播,转载的公众号无数。此论不细看,貌似...
盼孩子成绩好不如盼孩子性格好 ... 许多父母以为,家庭教育的核心就是让孩子学习好,因而掠夺式的开发孩子的智力,两三岁背唐诗,四五岁学英语...
柳州最新学区划分,最新或202... 柳州公办小学招生范围按照义务教育免试就近入学原则,市区公办小学实行依街道划片招生。本文为您介绍柳州小...
桂林最新学区划分,最新或202... 桂林公办小学招生范围按照义务教育免试就近入学原则,市区公办小学实行依街道划片招生。本文为您介绍桂林小...
面试热点题:回溯算法 电话号码... 前言: 如果你一点也不了解什么叫做回溯算法,那么推荐你看看这一篇回溯入门...
JetsonNano搭载的扩展... 在DIY机器人时,我们可以使用远程连接到机器人查看相关信息,也就是前面提...
贺州最新学区划分,最新或202... 最新或2023(历届)1月26日,教育部办公厅下发了《关于做好最新或2023(历届)城市义务教育招生...
百色最新学区划分,最新或202... 从百色右江区教育局了解到,百色右江区小学划片政策已经公布。太阳教育网为大家准备了最新或2023(历届...
渗透学习-CTF篇-web-C... 文章目录前言web入门部分反序列化web254web255web256web257web258 前...
梧州最新学区划分,最新或202... 今年,梧州市万秀区进行了秋季学期小学招生改革,首次采用“学区制”的形式进行招生。近日,有家长提出自己...
贵港最新学区划分,最新或202... 入学对象和安排顺序入学对象基本要求1.小学新生入学对象是城区户籍(包括符合进城务工人员随迁子女条件的...
玉林最新学区划分,最新或202... 江南区古定中心小学,是一所全日制公办小学,始建于1907年,该校于2002年获得了“中国名校”称号。...
钦州最新学区划分,最新或202... 钦州市区最新或2023(历届)初中学区划分方案根据《中华人民共和国义务教育法》《广西壮族自治区实施〈...
温州最新学区划分,最新或202... 温州公办小学招生范围按照义务教育免试就近入学原则,市区公办小学实行依街道划片招生。本文为您介绍温州小...
绍兴最新学区划分,最新或202... 绍兴公办小学招生范围按照义务教育免试就近入学原则,市区公办小学实行依街道划片招生。本文为您介绍绍兴小...
金华最新学区划分,最新或202... 金华公办小学招生范围按照义务教育免试就近入学原则,市区公办小学实行依街道划片招生。本文为您介绍金华小...
湖州最新学区划分,最新或202... 学校名称划片区域湖师附小教育集团(幸福里校区、余家漾校区、西山漾校区)潜庄公寓、白漾港小区、米兰花园...
PHP操作文件和目录 PHP操作文件和目录一、目录处理1.1 目录信息查询1.2 目录操作二、文件处理2.1 查询文件信息...
小白开发微信小程序20--we... 1、什么是SwaggerSwagger 项目已于 2015 年捐赠给 OpenAPI 计划ÿ...