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. }

目录
相关文章
|
2月前
|
算法
【优选算法】—— 字符串匹配算法
【优选算法】—— 字符串匹配算法
|
22天前
|
算法
代码随想录算法训练营第五十五天 | LeetCode 583. 两个字符串的删除操作、72. 编辑距离、编辑距离总结
代码随想录算法训练营第五十五天 | LeetCode 583. 两个字符串的删除操作、72. 编辑距离、编辑距离总结
24 1
|
26天前
|
存储 算法
图解Kmp算法——配图详解(超级详细)
图解Kmp算法——配图详解(超级详细)
|
1月前
|
算法 测试技术 C#
【字典树】【KMP】【C++算法】3045统计前后缀下标对 II
【字典树】【KMP】【C++算法】3045统计前后缀下标对 II
|
1月前
|
算法
KMP算法 与 strstr()函数
KMP算法 与 strstr()函数
|
1月前
|
算法
【算法学习--字符串】(不含KMP算法)
【算法学习--字符串】(不含KMP算法)
|
1天前
|
存储 算法 数据可视化
基于harris角点和RANSAC算法的图像拼接matlab仿真
本文介绍了使用MATLAB2022a进行图像拼接的流程,涉及Harris角点检测和RANSAC算法。Harris角点检测寻找图像中局部曲率变化显著的点,RANSAC则用于排除噪声和异常点,找到最佳匹配。核心程序包括自定义的Harris角点计算函数,RANSAC参数设置,以及匹配点的可视化和仿射变换矩阵计算,最终生成全景图像。
|
1天前
|
算法 Serverless
m基于遗传优化的LDPC码NMS译码算法最优归一化参数计算和误码率matlab仿真
MATLAB 2022a仿真实现了遗传优化的归一化最小和(NMS)译码算法,应用于低密度奇偶校验(LDPC)码。结果显示了遗传优化的迭代过程和误码率对比。遗传算法通过选择、交叉和变异操作寻找最佳归一化因子,以提升NMS译码性能。核心程序包括迭代优化、目标函数计算及性能绘图。最终,展示了SNR与误码率的关系,并保存了关键数据。
11 1
|
2天前
|
算法 调度
考虑需求响应的微网优化调度模型【粒子群算法】【matlab】
考虑需求响应的微网优化调度模型【粒子群算法】【matlab】
|
2天前
|
运维 算法
基于改进遗传算法的配电网故障定位(matlab代码)
基于改进遗传算法的配电网故障定位(matlab代码)