KMP算法(字符串匹配)(AcWing)

简介: KMP算法(字符串匹配)(AcWing)

KMP算法常用于字符串匹配,在匹配介绍KMP算法之前,先介绍如何暴力地匹配字符串

对两个字符串,用两个指针依次比较,代码:

1. for (int i = 1; i <= n; i ++ )
2. {
3. bool flag = true;
4. for (int j = 1; j <= m; j ++ )
5.     {
6. if (s[i + j - 1] != p[j])
7.         {
8.             flag=false;
9. break;
10.         }
11.     }
12. }

如果不匹配,相当于将短的那个字符串向右移动一位继续匹配,但这样依次比较的效率是非常低的

而KMP则是利用已经匹配好的字符串这个有效信息来减少重复的匹配

例如有图中,当长串和短串匹配成功了一段区间之后,在图中i和j+1的位置匹配失败了,按照常规思路我们是需要将短串向后移动一个位置继续重新开始匹配,但kmp就是利用好已经匹配好了的信息来减少匹配次数,就是令j=ne[j],从ne[j]的位置开始匹配,因为在图中我们用黑线画的部分其实是等效的,所以这一部分我们是不需要去匹配的,那么如何求这个ne[j]数组呢

这里要引入一个概念,就是前缀后缀

现在思路我们知道了,那么如何用代码来实现呢?

思路:

代码:

1.  for (int i = 2, j = 0; i <= m; i++)
2.  {
3.    while (j && str1[i] != str1[j + 1])j = ne[j];
4.    if (str1[j + 1] == str1[i])j++;
5.      ne[i] = j;
6.  }

接下来来模拟一个样例:

题目:

AC代码:

1. #include<iostream>
2. #include<cstring>
3. using namespace std;
4. const int N = 1000010, M = 10010;
5. char str[N], str1[M];
6. int ne[N], n, m;
7. 
8. int main(void)
9. {
10.   cin >>(str + 1) >>(str1 + 1);
11.   int n = strlen(str+1), m = strlen(str1+1);
12.   //获取ne数组
13.   for (int i = 2, j = 0; i <= m; i++)
14.   {
15.     while (j && str1[i] != str1[j + 1])j = ne[j];
16.     if (str1[j + 1] == str1[i])j++;
17.     ne[i] = j;
18.   }
19.   //开始匹配
20.   for (int i = 1, j = 0; i <= n; i++)
21.   {
22.     while (j && str1[j + 1] != str[i])j = ne[j];
23.     if (str[i] == str1[j + 1])j++;
24.     if (j == m)
25.     {
26.       printf("%d\n", i - m+1);
27.     }
28.   }
29.   for (int i = 1; i <= m; i++)
30.   {
31.     printf("%d ", ne[i]);
32.   }
33.   return 0;
34. }

目录
相关文章
|
4月前
|
算法
两个字符串匹配出最长公共子序列算法
本文介绍了最长公共子序列(LCS)问题的算法实现,通过动态规划方法求解两个字符串的最长公共子序列,并提供了具体的编程实现细节和示例。
135 1
两个字符串匹配出最长公共子序列算法
|
4月前
|
算法
第四章 KMP算法理论基础
第四章 KMP算法理论基础
40 0
|
4月前
|
算法
KMP算法
KMP算法
54 0
|
6月前
|
算法 C++
A : DS串应用–KMP算法
这篇文章提供了KMP算法的C++实现,包括计算模式串的next数组和在主串中查找模式串位置的函数,用于演示KMP算法的基本应用。
|
6月前
|
算法 Java
掌握算法学习之字符串经典用法
文章总结了字符串在算法领域的经典用法,特别是通过双指针法来实现字符串的反转操作,并提供了LeetCode上相关题目的Java代码实现,强调了掌握这些技巧对于提升算法思维的重要性。
|
6月前
|
算法
KMP算法
KMP算法
48 0
|
2天前
|
算法 数据安全/隐私保护 计算机视觉
基于FPGA的图像双线性插值算法verilog实现,包括tb测试文件和MATLAB辅助验证
本项目展示了256×256图像通过双线性插值放大至512×512的效果,无水印展示。使用Matlab 2022a和Vivado 2019.2开发,提供完整代码及详细中文注释、操作视频。核心程序实现图像缩放,并在Matlab中验证效果。双线性插值算法通过FPGA高效实现图像缩放,确保质量。
|
1月前
|
算法 数据安全/隐私保护 计算机视觉
基于Retinex算法的图像去雾matlab仿真
本项目展示了基于Retinex算法的图像去雾技术。完整程序运行效果无水印,使用Matlab2022a开发。核心代码包含详细中文注释和操作步骤视频。Retinex理论由Edwin Land提出,旨在分离图像的光照和反射分量,增强图像对比度、颜色和细节,尤其在雾天条件下表现优异,有效解决图像去雾问题。
|
1月前
|
算法 数据可视化 安全
基于DWA优化算法的机器人路径规划matlab仿真
本项目基于DWA优化算法实现机器人路径规划的MATLAB仿真,适用于动态环境下的自主导航。使用MATLAB2022A版本运行,展示路径规划和预测结果。核心代码通过散点图和轨迹图可视化路径点及预测路径。DWA算法通过定义速度空间、采样候选动作并评估其优劣(目标方向性、障碍物距离、速度一致性),实时调整机器人运动参数,确保安全避障并接近目标。
147 68
|
1月前
|
算法 数据安全/隐私保护
室内障碍物射线追踪算法matlab模拟仿真
### 简介 本项目展示了室内障碍物射线追踪算法在无线通信中的应用。通过Matlab 2022a实现,包含完整程序运行效果(无水印),支持增加发射点和室内墙壁设置。核心代码配有详细中文注释及操作视频。该算法基于几何光学原理,模拟信号在复杂室内环境中的传播路径与强度,涵盖场景建模、射线发射、传播及接收点场强计算等步骤,为无线网络规划提供重要依据。