kmp算法c++

kmp算法通过next数组使查找失败时减少跳转后比较的次数来优化字符串查找,next数组存储了前缀和后缀相同的位置信息,类似动规,可以存储查找数组的信息来防止重复查找,最终复杂度可以达到O(n+m)。

以t=“abcabd”字符串为例进行讲解

next数组存储的是当前位置与前面具有相同前缀的长度,同时在查找失败后,可以在失败位置的其哪一个位置对应的next数组值来跳转到具有相同前缀的字符串的下一个位置,这样就可以避免重复验证前缀部分是否和被查找的字符串匹配。

具体的演示可以看和手算方法见「六分钟速通」KMP算法求解流程,up讲的简单明了。视频里pm即为文中的next数组

next[i]里的数字代表包括t[i]在内的部分与前面字符串部分子串相同的长度,同时该数字也是前面相同子串的下一个字符的下标。

这里设j,k作为下标表示,j为当前计算的下标,k为前面与当前计算下标j结尾的子串相同的子串的结尾下标,即具有相同前缀的子串结尾下标。

当k=0时,有两种情况,一是t[j]=t[k],那么此时next[j]=next[j-1]+1,即相同前缀部分变大,如果不同,此时k=0,则next[j]应该为0,即没有相同前缀。

当k!=0时,如果t[k]=t[j],则按上述计算方法来表示扩大相同前缀,如果t[k]!=t[j],那么向前找相同前缀,做法时k=next[k-1],因为next数组存储的是当前位置与前面具有相同前缀的长度,则向前查找只需要找到以k-1结尾的相同子串的长度,如果还不同,重复向前查找,直到k=0或找到相同前缀为止。

#include <iostream>
#include <string.h>
#include <stdlib.h>
using namespace std;void grtnext(char *c, int *nextKMP)
{int len = strlen(c);nextKMP[0] = 0;int k = 0;int j = 1;while (j < len){if (k == 0){if (c[j] == c[k]){nextKMP[j] = 1;k++;}elsenextKMP[j] = 0;j++;continue;}if (c[j] == c[k]){nextKMP[j] = nextKMP[j - 1] + 1;k++;j++;}else{k = nextKMP[k - 1];}}
}
int KMP(char *s, char *t, int *next, int *re)
{int lens = strlen(s);int lent = strlen(t);int i = 0, j = 0;while (i < lens && j < lent){re[0]++;if (s[i] == t[j]){i++;j++;}else{if (j == 0)i++;elsej = next[j - 1];}}if (j == lent)return i - lent;return -1;
}
int trlen(char *s)
{int len = 0;while (s[len] != '\0'){len++;}
}
int main()
{char *c = "aabaac";char *s = "aabaabaaac";int len = strlen(c);int re[2] = {0};int *nextKMP = new int[len];grtnext(c, nextKMP);int k = KMP(s, c, nextKMP, re);// for (int i = 0; i < len; i++)// {//     cout << nextKMP[i] << " ";// }
if(k==-1)cout<<"no find"<<endl;
elsecout << k + 1 << endl;cout << re[0]; // 输出比较次数system("pause");
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/25345.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

车载电子电气架构 --- 车载信息安全

我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 屏蔽力是信息过载时代一个人的特殊竞争力,任何消耗你的人和事,多看一眼都是你的不对。非必要不费力证明自己,无利益不试图说服别人,是精神上的节…

Vite - 开发初体验,以及按需导入配置

目录 开始 创建一个 Vite 项目 项目结构 /src/main.js index.html package.json vite.config.js Vite 项目中使用 vue-router Vite 组件的“按需引入” 传统的方式引入一个组件 传统方式引入带来的问题 解决办法&#xff08;配置 按需引入 插件&#xff09; 示例&…

嵌入式学习——Linux高级编程复习(线程)——day40

1. 线程 1.1 定义 线程是一个轻量级的进程 是一个任务被创建、调度、消亡的过程 1.2 线程和进程的区别与联系 1. 线程是CPU任务调度的最小单元 2. 进程是操作系统资源分配的最小单元 3. 线程&#xff08;Thread&#xff09;是操作系统能够进行运算调度的最小单位…

OpenFeign远程接口调用使用公共模块出现的错误

今天在使用openfeign和sentinel实现fallback服务降级时遇到找不到类型的异常 检查代码发现没有错误&#xff0c;EnableFeignClients也在启动类上标注了 错误信息&#xff1a;A component required a bean of type com.zxc.cloud.apis.PayFeignSentinelApi that could not be f…

《精通ChatGPT:从入门到大师的Prompt指南》第9章:实战练习

第9章&#xff1a;实战练习 9.1 Prompt练习题 在本节中&#xff0c;我们将提供一系列练习题&#xff0c;旨在帮助读者通过实际操作提升使用ChatGPT的能力。这些练习题涵盖了从基础到高级的不同难度级别&#xff0c;并针对各种应用场景设计&#xff0c;确保读者能够在实际使用…

基于Springboot + vue实现的火锅店管理系统

作者主页&#xff1a;Java码库 主营内容&#xff1a;SpringBoot、Vue、SSM、HLMT、Jsp、PHP、Nodejs、Python、爬虫、数据可视化、小程序、安卓app等设计与开发。 收藏点赞不迷路 关注作者有好处 文末获取源码 技术选型 【后端】&#xff1a;Java 【框架】&#xff1a;spring…

私有云和多云管理平台 | Cloudpods v3.11.4 正式发布

本次 3.11.4 更新亮点为&#xff1a;系统镜像引入社区镜像&#xff0c;用户可以一键导入各主流开源操作系统镜像&#xff0c;方便用户上手使用。持续迭代共享 LVM&#xff0c;支持快照&#xff0c;主备机等特性&#xff0c;修复迁移删除镜像缓存等 BUG。 功能优化 【费用】费…

【调度算法】Boltzmann选择

Boltzmann选择是一种基于Boltzmann分布的选择策略&#xff0c;主要用于进化算法中的个体选择过程。它通过模拟物理系统的热平衡状态来调节个体选择的概率&#xff0c;能够在进化初期保持种群多样性&#xff0c;并在进化后期集中选择适应度高的个体。 Boltzmann选择的表达式 B…

EON安装ASE Interface

安装 测试系统ubuntu。如果你python2和python3总是纠缠不清&#xff0c;可以sudo apt install python-is-python3直接解决。 经检查&#xff0c;我PC的 python地址为&#xff1a; /usr/include/python3.8/ pybind11地址为&#xff1a; /usr/include/pybind11/ 已确认python…

web前端的MySQL:跨领域之旅的探索与困惑

web前端的MySQL&#xff1a;跨领域之旅的探索与困惑 在数字化浪潮的推动下&#xff0c;web前端与MySQL数据库似乎成为了两个不可或缺的领域。然而&#xff0c;当我们将这两者放在一起&#xff0c;尝试探索web前端与MySQL之间的交互与关联时&#xff0c;却发现这是一次充满困惑…

AE电源pinnacle软件新款老款二款软件

AE电源pinnacle软件新款老款二款软件

C++从入门到精通(最详细教程,12万总结,带你掌握c++知识,涵盖大量知识点)

目录 一、面向对象的思想 二、类的使用 1.类的构成 2.类的设计 三、对象的基本使用 四、类的构造函数 1.构造函数的作用 2.构造函数的特点 3.默认构造函数 3.1.合成的默认构造函数 3.2.手动定义的默认构造函数 四、自定义的重载构造函数 五、拷贝构造函数 1.手动…

联合体和枚举<C语言>

导言 在C语言中除了结构体外&#xff0c;联合体和枚举也是自定义类型&#xff0c;联合体主要用于节省空间&#xff0c;在同一块内存存储多种类型的数据&#xff0c;而枚举可以提高代码的可读性、可维护性。 联合体&#xff08;union&#xff09; 它还有个更容易理解的名字&…

Rust-08-枚举和模式匹配

枚举类 结构体给予你将字段和数据聚合在一起的方法&#xff0c;像 Rectangle 结构体有 width 和 height 两 个字段。而枚举给予你将一个值成为一个集合之一的方法。比如&#xff0c;我们想让 Rectangle 是一 些形状的集合&#xff0c;包含 Circle 和 Triangle 。为了做到这个&…

硬件工程师需要掌握的工具

软件工具&#xff1a; 一、常用画图软件 1、AD/protel 简单好学&#xff0c;在低端市场使用的比较多。建议刚进入硬件工程师岗位或者大学生使用。 2、Candence/allegro Candence画复杂的板子相对更加有优势。但是学习难度比较高&#xff0c;但是如果学会AD后&#xff0c;可…

MSPM0——GPIO的使用

在dl_gpio.h库函数文件中&#xff0c;有三个函数可以控制引脚状态。 __STATIC_INLINE void DL_GPIO_setPins(GPIO_Regs* gpio, uint32_t pins) 该函数为控制引脚输出高电平&#xff0c;按照本例程中LED的引脚&#xff0c;则可以写为 DL_GPIO_setPins(LED1_PORT,LED1_PIN_14_…

适用于电脑的 5 大嗨格式数据恢复替代方案

嗨格式数据恢复是有一定知名度的 Windows 和 Mac 恢复程序&#xff0c;旨在恢复格式化、删除和丢失的图片、视频和音频。该应用程序支持多种文件格式以及相机 RAW 图像。最好的部分&#xff1f;它的预览功能可以在恢复照片和其他媒体文件之前检查和验证它​​们——这可以节省大…

番外篇 | 超越ReLU却鲜为人知,YOLOv5改进之崛起的最佳激活函数GELU!

前言:Hello大家好,我是小哥谈。作为决定神经网络是否传递信息的「开关」,激活函数对于神经网络而言至关重要。不过今天被人们普遍采用的ReLU真的是最高效的方法吗?最近在社交网络上,人们找到了一个看来更强大的激活函数:GELU,这种方法早在2016年即被人提出,然而其论文迄…

快排(快速排序)的递归与非递归实现(文末附完整代码)

快排有几种不同的写法&#xff0c;下面一一来介绍并实现。其中又分为递归和非递归的写法&#xff0c;但大体思路相同&#xff0c;只是代码实现略有不同。(注&#xff1a;文章中的完整代码中&#xff0c;Swap()函数均省略未写&#xff0c;记得自己补充) 递归写法 递归的写法类…

glm-4v-9b 部署

glm-4v-9b 模型文件地址 GLM-4 仓库文件地址 官方测试 硬件配置和系统要求 官方测试硬件信息: OS: Ubuntu 22.04Memory: 512G…