字符串匹配算法(BF&&KMP)

这篇具有很好参考价值的文章主要介绍了字符串匹配算法(BF&&KMP)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

个人主页:平行线也会相交
欢迎 点赞👍 收藏✨ 留言✉ 加关注💓本文由 平行线也会相交 原创
收录于专栏【数据结构初阶(C实现)】
字符串匹配算法(BF&&KMP)

字符串匹配算法

在学习这个算法之前,我们先来看看什么时字符串匹配算法,简单来说有一个主串和一个子串,查找子串在主串的位置,然后返回这个位置的下标。
想要实现这个功能其实有很多方法,比较有名的算法有两种:一种是BF算法又称暴力算法,另一种就是KMF算法。

BF算法

BF算法:思想就是将目标串S的第一个字符与模式串T的第一个字符进行匹配,如果相等,则继续比较S的第二个字符和T的第二个字符;如果不相等,则比较S的第二个字符和T的第一个字符,依次比较下去,直到得出最后的酦醅结果。
举个例子:
字符串匹配算法(BF&&KMP)

代码实现

#define _CRT_SECURE_NO_WARNINGS 1
//BF算法

#include<assert.h>

#include<stdio.h>

//str为主串,sub为子串
int BF(char* str, char* sub)
{
	assert(str != NULL && sub != NULL);
	if (str == NULL || sub == NULL)
		return -1;
	int lenStr = strlen(str);
	int lenSub = strlen(sub);
	int i = 0;
	int j = 0;
	while (i < lenStr && j < lenSub)
	{
		if (str[i] == sub[j])
		{
			i++;
			j++;
		}
		else
		{
			i = i - j + 1;
			j = 0;
		}
	}
	if (j >= lenSub)//如果j>=lenSub说明子串遍历完成,即匹配成功,返回i的下标。
	{
		return i - j;
	}
	//不存在直接返回-1
	return -1;
}

int main()
{
	printf("%d\n", BF("ababcabcdabcde", "abcd"));
	printf("%d\n", BF("ababcabcdabcde", "abcdf"));
	printf("%d\n", BF("ababcabcdabcde", "ab"));
	return 0;
}

字符串匹配算法(BF&&KMP)

KMP算法

KMP算法就是对BF算法是一种对BF算法的改进,该算法核心就是可以利用匹配失败后的信息,尽量减少模式串与字串的匹配次数以到达快速匹配的目的(具体shi)。
KMP与BF算法的区别就是KMP算法主串的并不会回退;并且j不会移动到0号位置,而是移动到一个特定的位置。
我们直接来举个例子:

字符串匹配算法(BF&&KMP)

此时ij位置的字符不匹配了。此时i是不进行回溯的,而是要对j进行回溯,那么j应该回溯到哪个位置呢?
字符串匹配算法(BF&&KMP)
由于每个位置要回溯的位置可能不一样,所以就引出了next数组。即用next[j]=k来表示。不同的j对应一个K值。这个K就是将来j要进行回溯的位置。如上图我们求的是当j=5的时候,K的值就是2,即将来j要回溯到下标为2的位置。即next[5]=2;。再比如说,当j是4的时候,K的值就是1,即next[4]=1;
字符串匹配算法(BF&&KMP)

关于K值求取的规则如下:

1.找到匹配成功部分的两个相等的真串(不包含本身),一个以下标0开始,另一个j-1下标结束。
2.无论是什么数据,如果我们是从0开始计数(这里按照数组下标从0开始的习惯所以从0开始计数),那么next[0]=-1;next[1]=0;如果我们从1开始计数,那么next[0]=0;next[1]=1

来练习一下
"a b a b c a b c d a b c d e ",求其next数组
答案如下图:
字符串匹配算法(BF&&KMP)

代码实现

#include<stdio.h>
#include<assert.h>
#include<stdlib.h>
#include<string.h>

void GetNext(char* sub, int* next, int lenSub)
{
	next[0] = -1;
	next[1] = 0;
	int i = 2;
	int k = 0;
	while (i < lenSub)
	{
		if (k == -1 || sub[i - 1] == sub[k])
		{
			next[i] = k + 1;
			i++;
			k++;
		}
		else
		{
			k = next[k];
		}
	}
}

int KMP(char* str, char* sub, int pos)
{
	assert(str != NULL && sub != NULL);
	int lenStr = strlen(str);
	int lenSub = strlen(sub);
	if (lenStr == 0 || lenSub == 0)
		return -1;
	if (pos < 0 || pos >= lenStr)
		return -1;

	int* next = (int*)malloc(sizeof(int) * lenSub);
	assert(next != NULL);

	GetNext(sub, next, lenSub);

	int i = pos;//遍历主串
	int j = 0;//遍历子串

	while (i < lenStr && j < lenSub)
	{
		if (j == -1 || str[i] == sub[j])
		{
			i++;
			j++;
		}
		else
		{
			j = next[j];
		}
	}
	if (j >= lenSub)
	{
		return i - j;
	}
	return -1;
}

int main()
{

	printf("%d\n", KMP("ababcabcdabcde", "abcd", 0));
	printf("%d\n", KMP("ababcabcdabcde", "abcdf", 0));
	printf("%d\n", KMP("ababcabcdabcde", "ab", 0));
	return 0;
}

字符串匹配算法(BF&&KMP)

nextval数组改进

下面来看nextval数组的求解规则

1.无论是什么数据,nextval[0]=-1;(这里还是默认数组的习惯从0开始计数)。如果是从1开始计数,则nextval[0]=0;
2.从第二位开始,我们用next[i]值对应的字符i值对应的字符进行比较。如果相等,则nextval[i]就等于next[i]值对应字符的nextval[i]值;如果不相等,则nextval[i]值就等于当前字符对应的next值

我们还是来进行举例:

求模式串"a b c a a b b c a b c a a b d a b"
字符串匹配算法(BF&&KMP)
下面来看详细过程:
个字符a对应的nextval[0]一定为-1(按照从0开始计数的话)。即nextval[0]=-1;
个字符b的next值即next[1]=0;所以第二个字符和下标为0的字符进行比较。发现不相等,所以nextval[1]=第二个字符所对应的next值,即nextval[1]=0;
个字符c的next值即next[2]=0;所以第三个字符和下标为0的字符进行比较。发现不相等,所以nextval[2]=第三个字符所对应的next值,即nextval[2]=0;
个字符a的next值即next[3]=0;所以第四个字符和下标为0的字符进行比较。发现相等了,所以nextval[3]=下标为0的字符所对应的nextval值,在这里就是nextval[3]=nextval[0]
个字符a的next值即next[4]=1;所以第五个字符a和下标为1的字符b进行比较。发现不相等,所以nextval[4]=当前字符(即指的是第五个字符)所对应的next值,所以最终nextval[4]=next[4]=1
依此类推进行分析,所以最终该串的nextval数组就如上图所示。

好了,以上就是关于字符串BF和KMP算法的一个记录。
就到这里吧,各位,再见啦!!!
字符串匹配算法(BF&&KMP)文章来源地址https://www.toymoban.com/news/detail-415838.html

到了这里,关于字符串匹配算法(BF&&KMP)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • C#,字符串匹配(模式搜索)KMP算法的源代码与数据可视化

      D.E.Knuth   J.H.Morris KMP 算法(Knuth-Morris-Pratt 算法)是其中一个著名的、传统的字符串匹配算法,效率比较高。 KMP算法由 D.E.Knuth , J.H.Morris 和 V.R.Pratt 在 Brute-Force 算法的基础上提出的模式匹配的改进算法。因此人们称它为“克努特—莫里斯—普拉特算法”,简称KMP算法。K

    2024年01月25日
    浏览(29)
  • P3375 【模板】KMP 字符串匹配

    给出两个字符串 s 1 s_1 s 1 ​ 和 s 2 s_2 s 2 ​ ,若 s 1 s_1 s 1 ​ 的区间 [ l , r ] [l, r] [ l , r ] 子串与 s 2 s_2 s 2 ​ 完全相同,则称 s 2 s_2 s 2 ​ 在 s 1 s_1 s 1 ​ 中出现了,其出现位置为 l l l 。 现在请你求出 s 2 s_2 s 2 ​ 在 s 1 s_1 s 1 ​ 中所有出现的位置。 定义一个字符串 s s s 的

    2024年02月14日
    浏览(29)
  • PTA 7-1 字符串模式匹配(KMP)

    给定一个字符串 text 和一个模式串 pattern,求 pattern 在text 中的出现次数。text 和 pattern 中的字符均为英语大写字母或小写字母。text中不同位置出现的pattern 可重叠。 输入格式: 输入共两行,分别是字符串text 和模式串pattern。 输出格式: 输出一个整数,表示 pattern 在 text 中的出

    2024年02月06日
    浏览(22)
  • 【leetcode 力扣刷题】字符串匹配之经典的KMP!!!

    以下是能用KMP求解的算法题,KMP是用于字符串匹配的经典算法【至今没学懂………啊啊啊】 题目链接:28. 找出字符串中第一个匹配项的下标 题目内容: 题意还是很好理解的,要在字符串haystack中查找一个完整的needle,即字符串匹配。 暴力求解就是用 两层循环 :haystack从第

    2024年02月09日
    浏览(23)
  • Leetcode的AC指南 —— 字符串/KMP:28.找出字符串中第一个匹配项的下标

    摘要: Leetcode的AC指南 —— 字符串/KMP:28.找出字符串中第一个匹配项的下标 。题目介绍:给你两个字符串 haystack 和 needle ,请你在 haystack 字符串中找出 needle 字符串的第一个匹配项的下标(下标从 0 开始)。如果 needle 不是 haystack 的一部分,则返回 -1 。 题目介绍 :给你两

    2024年02月02日
    浏览(37)
  • 算法基础(一):串匹配问题(BF,KMP算法)

    好家伙,学算法, 这篇看完,如果没有学会KMP算法,麻烦给我点踩 希望你能拿起纸和笔,一边阅读一边思考,看完这篇文章大概需要(20分钟的时间)   我们学这个算法是为了解决 串匹配 的问题 那什么是串匹配? 举个例子: 我要在\\\"彭于晏 吴彦祖 \\\"这段字符串中找到\\\" 吴彦祖 \\\"字符串 这

    2024年02月08日
    浏览(59)
  • 数据结构--字符串的KMP算法

    朴素模式匹配算法: 一旦发现当前这个子串中某个字符不匹配,就只能转而匹配下一个子串(从头开始) 但我们可以知道: 不匹配的字符之前,一定是和模式串一致的 color{red}不匹配的字符之前,一定是和模式串一致的 不匹配的字符之前,一定是和模式串一致的 我们可以利用

    2024年02月12日
    浏览(35)
  • LeetCode:459. 重复的子字符串 —【2、KMP算法】

    🍎道阻且长,行则将至。🍓 🌻算法,不如说它是一种思考方式🍀 算法专栏: 👉🏻123 题目描述 :给定一个非空的字符串 s ,检查是否可以通过由它的一个子串重复多次构成。 来源:力扣(LeetCode) 难度: 简单 提示: 1 = s.length = 104 s 由小写英文字母组成 示例 1: 输入:

    2024年02月04日
    浏览(32)
  • 《数据结构》实验报告四:串的模式匹配(BF算法、KMP算法)

    1、了解 串 的基本概念。 2、掌握 串的模式匹配算法 的实现 。 说明以下概念 1、模式匹配:          串的模式匹配就是 子串的定位运算 。          设有两个字符串 S 和 T ,S为 主串(正文串) ,T为 子串(模式串) 。在主串S中查找与模式串T相匹配的子串,若匹配成功,确定

    2024年02月01日
    浏览(32)
  • 代码随想录 Leetcode459. 重复的子字符串(KMP算法)

            此解法读者需要了解什么是KMP算法以及KMP算法中next数组的具体含义才能理解         因为在KMP算法的next数组中,next[index]表示 i ndex之前的最大长度的相同前缀后缀值 ,那么要判断整个字符串中是否由重复字串构成,只需要以下两个条件:         1.next[n - 1] !=

    2024年01月19日
    浏览(43)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包