数据结构(KMP算法)

//KMP求next数组的详细步骤

//1 申请一个next数组 长度和字串长度一致

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

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

//4 再申请一个变量NEX 用来保存当前字符的失配时 可以看到的最长公共前后缀的共有长度(换句话说就是可以有多少个字符可以不用比较)

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

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

//7 如果相同 则让NEX+1 然后赋值给下一个字符的next数组位置 (相当于将原本此时的右绿和左绿线末尾又各自加了一个相同的字符 所以右绿和左绿的共有长度+1)

//8 如果不相同 则让NEX去保存其回退位置字符的回退位置 接着比较 直到比较成功 或者NEX触底(NEX==-1)

//9 如果一直比较失败 也就是NEX==-1,则此时将NEX+1变成0赋值给下一个字符的next数组的位置

//10 当while循环结束 则next数组全部赋值完成

//获取next数组

int* Get_Next(const char* SubStr)

{

//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;

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

while (j < +1 < strlen(SubStr))

{

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

//7 如果相同 则让NEX+1 然后赋值给下一个字符的next数组位置 (相当于将原本此时的右绿和左绿线末尾又各自加了一个相同的字符 所以右绿和左绿的共有长度+1)

//9 如果一直比较失败 也就是NEX==-1,则此时将NEX+1变成0赋值给下一个字符的next数组的位置

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

{

next++j = ++NEX;

}

//8 如果不相同 则让NEX去保存其回退位置字符的回退位置 接着比较 直到比较成功 或者NEX触底(NEX==-1)

else

{

NEX = nextNEX;

}

//10 当while循环结束 则next数组全部赋值完成

return next;

}

}

//next数组再次优化 得到next_well数组

//1 先搞出来next数组

//2 在申请一个nextval数组 并将其第一个格子赋值为-1

//3 申请变量j 让其从第二个字符开始向后遍历

//4 如果j指向的字符和next数组变成的回退位置的字符相同 则说明next数组保存的回退位置无效 直接将其回退位置字符的nextval对应值照抄过来

//5 如果j指向的字符和next数组变成的回退位置的字符不相同 则说明next数组保存的回退位置有效 直接照抄过来即可

int* Get_Nextval(const char* SubStr)

{

//1 先搞出来next数组

int* next = Get_Next(SubStr);

//2 在申请一个nextval数组 并将其第一个格子赋值为-1

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

if (NULL == nextval)

exit(EXIT_FAILURE);

nextval0 = -1;

int len1 = strlen(SubStr);

//3 申请变量j 让其从第二个字符开始向后遍历

for (int j = 1; j < len1; j++)

{

//5 如果j指向的字符和next数组变成的回退位置的字符不相同 则说明next数组保存的回退位置有效 直接照抄过来即可

if (SubStrnextval\[j] != SubStrnext\[j])

{

nextvalj=nextj;

}

//4 如果j指向的字符和next数组变成的回退位置的字符相同 则说明next数组保存的回退位置无效 直接将其回退位置字符的nextval对应值照抄过来

else

{

nextvalj = nextvalnext\[j];

}

}

return nextval;

}

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

{

assert(MainStr != NULL && SubStr != NULL);

int i = 0;

int j = 0;

int mainlen = strlen(MainStr);

int SubStrlen = strlen(SubStr);

int* next = Get_Next(SubStr);

while (i < (int)mainlen && j < (int)SubStrlen)

{

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

{

i++;

j++;

}

else

{

j = nextj;

}

}

if (j < SubStrlen)

{

return -1;

}

return i - j;

}

相关推荐
南境十里·墨染春水1 天前
C++ 工厂模式:从入门到进阶,彻底掌握对象创建的艺术
开发语言·c++·算法
@insist1231 天前
系统架构设计师-实时性评价、调度算法与内核架构选型
算法·架构·系统架构·软考·系统架构设计师·软件水平考试
一只齐刘海的猫1 天前
【Leetcode】找到字符串中所有字母异位词
算法·leetcode·职场和发展
海清河晏1111 天前
数据结构 | 八大排序
数据结构·算法·排序算法
liulilittle1 天前
固定数组时间轮的槽过载优化:桶链表与批次执行
网络·数据结构·链表
IronMurphy1 天前
【算法五十七】146. LRU 缓存
算法·缓存
Irissgwe1 天前
数据结构-栈和队列
数据结构·c++·c·栈和队列
两片空白1 天前
数据容器集合set/frozenset
数据结构
凌波粒1 天前
LeetCode--108.将有序数组转换为二叉搜索树(二叉树)
算法·leetcode·职场和发展