2026.5.24 数据结构 KMP算法实现

#define _CRT_SECURE_NO_WARNINGS

#include <stdio.h>

#include <stdlib.h>

#include <string.h>

#include <assert.h>

#include <memory.h>

int BF_Search(const char* MainStr, const char* SubStr)

{

//BF算法逻辑:

//1.申请两个指针i和j,分别指向主串和子串的开始位置

int i = 0;

int j = 0;

//"ABCD" strlen=4

//2.进入while循环,循环的条件是i和j还合法(换句话说就是还没有越界)

while (i < strlen(MainStr) && j < strlen(SubStr))

{

//3.比较i和j指向的字符,如果相同,则让i和j同步向后移动一个位置

if (MainStri == SubStrj)

{

i++;

j++;

}

//4.如果不相同,则让指向子串的j变量,回退到开始位置(0下标)

else

{

i = i - j + 1;

j = 0;

}

}

//5.如果while循环进不出,则说明查找结束了,只需要对j来做判断,就可以知道是否查找成功(j走出自身范围则找到,相反j未走出自身范围则没找到)

if (j < strlen(SubStr))

{

return -1;

}

return i - j;

}

//获取next数组

int* Get_Next(const char* SubStr)

{

//kmp求next数组的详细步骤(已知推未知) :

//1.申请一个next数组,长度和子串长度一致

int* next = (int*)malloc(strlen(SubStr) * sizeof(int));

if (NULL == next)

exit(EXIT_FAILURE);

//2.将next数组的前两个格子直接赋值 - 1和0

next0 = -1;

next1 = 0;

//3.申请一个变量j指向当前最新的已知其next值的字符(初始化为1号下标)

int j = 1;

//4.再申请一个变量NEX,用来保存当前字符的失配时,向前可以看到的最

// 长公共前后缀的共有长度(换句话说就是可以有多少个字符可以不用比较)

int NEX = nextj;//0

//5.进入while循环,循环条件是j合法(j最多指向倒数第二个字符)

while (j+1 < strlen(SubStr))

{

//9.如果一直比较失败,也就是NEX == -1了,则此时将NEX + 1变成0

// 赋值给下一个字符的next数组的位置

//6.用当前j指向的字符和其NEX保存的回退位置的字符进行比较

//7.如果相同,则让NEX + 1, 然后赋值给下一个字符的next数组位置

// (相当于将原本此时的左绿和右绿线末尾又各自了一个相同的字符,

// 所以左绿和右绿线共有长度 + 1)

if (NEX==-1 || SubStrj == SubStrNEX)

{

next++j = ++NEX;

}

//8.如果不相同,则让NEX去保存其回退位置字符的回退位置,接着比

// 较,直到比较成功,或者NEX触底(触底就是NEXT == -1)

else

{

NEX = nextNEX;

}

}

//10.当while循环退出,则next数组全部赋值完毕

return next;

}

int KMP_Search(const char* MainStr, const char* SubStr)

{

//BF算法逻辑:

//1.申请两个指针i和j,分别指向主串和子串的开始位置

int i = 0;

int j = 0;

//1.5 提供子串对应的next数组

int* next = Get_Next(SubStr);

//"ABCD" strlen=4

//2.进入while循环,循环的条件是i和j还合法(换句话说就是还没有越界)

while (i < (int)strlen(MainStr) && j < (int)strlen(SubStr))

{

//3.比较i和j指向的字符,如果相同,则让i和j同步向后移动一个位置

if (j == -1 || MainStri == SubStrj)

{

i++;

j++;

}

//4.如果不相同,则让指向子串的j变量,回退到开始位置(0下标)

else

{

//i打死不回退

j = nextj;//j回退到一个合适的位置

}

}

//5.如果while循环进不出,则说明查找结束了,只需要对j来做判断,就可以知道是否查找成功(j走出自身范围则找到,相反j未走出自身范围则没找到)

if (j < strlen(SubStr))

{

return -1;

}

return i - j;

}

//获取nextval数组

int* Get_Nextval(const char* SubStr)

{

int* next = Get_Next(SubStr);

int *nextval = (int*)malloc(strlen(SubStr) * sizeof(int));

if (NULL == nextval)

exit(EXIT_FAILURE);

nextval0 = -1;

for (int j = 1; j < strlen(SubStr); j++)

{

if (SubStrj != SubStrnext\[j])

nextvalj = nextj;

else

nextvalj = nextvalnext\[j];

}

free(next);

return nextval;

}

int KMP_Search2(const char* MainStr, const char* SubStr)

{

//BF算法逻辑:

//1.申请两个指针i和j,分别指向主串和子串的开始位置

int i = 0;

int j = 0;

//1.5 提供子串对应的next数组

int* nextval = Get_Nextval(SubStr);

//"ABCD" strlen=4

//2.进入while循环,循环的条件是i和j还合法(换句话说就是还没有越界)

while (i < (int)strlen(MainStr) && j < (int)strlen(SubStr))

{

//3.比较i和j指向的字符,如果相同,则让i和j同步向后移动一个位置

if (j == -1 || MainStri == SubStrj)

{

i++;

j++;

}

//4.如果不相同,则让指向子串的j变量,回退到开始位置(0下标)

else

{

//i打死不回退

j = nextvalj;//j回退到一个合适的位置

}

}

//5.如果while循环进不出,则说明查找结束了,只需要对j来做判断,就可以知道是否查找成功(j走出自身范围则找到,相反j未走出自身范围则没找到)

if (j < strlen(SubStr))

{

return -1;

}

return i - j;

}

int main()

{

/*const char* MainStr = "ABABCABCDABCDE";

const char* SubStr = "ABCD";*/

/*const char* MainStr = "AAAAAAAAAAB";

const char* SubStr = "AAAAAAAB";*/

//int index = BF_Search(MainStr, SubStr);

const char* MainStr = "ABCBAAABCDABCDE";

const char* SubStr = "ABCD";

//int index = KMP_Search(MainStr, SubStr);

int index = KMP_Search2(MainStr, SubStr);

if (index == -1)

{

printf("没找到\n");

}

else

{

printf("找到了,且在%d找到的\n", index);

}

return 0;

}

相关推荐
wabs66623 分钟前
关于动态规划【力扣1143.最长公共子序列的思考】
算法·leetcode·动态规划
剑挑星河月43 分钟前
54.螺旋矩阵
java·算法·leetcode·矩阵
Robot_Nav1 小时前
MPPI 局部规划器实验设计讲解
人工智能·算法·mppi
mingo_敏2 小时前
Mean-Teacher 均值教师自训练框架详解
算法·均值算法
星空露珠2 小时前
迷你世界UGc3.0脚本Wiki[剧情动画模块管理接口 Timeline]
开发语言·数据结构·算法·游戏·lua
笨笨没好名字2 小时前
Leetcode刷题python3版第一周(下)
linux·算法·leetcode
手写码匠2 小时前
手写 LLM 安全护栏:从内容审核到越狱防御的完整实现
人工智能·深度学习·算法·aigc
luj_17682 小时前
草酸与烟酸对消化及糖代谢的影响解析
服务器·c语言·开发语言·经验分享·算法
青风973 小时前
16-ADAPTRACK:基于自适应阈值的多目标跟踪匹配算法
人工智能·算法·目标跟踪
汤姆yu3 小时前
macOS系统下Aider完整安装、配置与实战使用教程
大数据·人工智能·算法·macos·github·copilot