关于KMP算法中next数组和nextVal数组求法的整理

目录

比较经典的例子:

next数组的求解方法是:

具体计算next数组过程如下:

nextval数组的求解方法:

具体计算nextval的过程如下:

练练手:


请读者静下心来,拿出笔和纸,慢慢地按照下面的思路走,最后就会明白了。

比较经典的例子:

位数
模式串abaabcac
next01122312
nextval01021302

next数组的求解方法是

  • 第一位的next值为0,第二位的next值为1,后面求解每一位的next值时,根据前一位进行比较。
  • 首先将前一位与其next值对应的内容进行比较,
    • 如果相等,则该位的next值就是前一位的next值加上1;
    • 如果不等:
      • 向前继续寻找next值对应的内容来与前一位进行比较,直到找到某个位上内容的next值对应的内容与前一位相等为止,则这个位对应的值加上1即为需求的next值;
      • 如果找到第一位都没有找到与前一位相等的内容,那么需求的位上的next值即为1。

具体计算next数组过程如下:

1. 前两位必定为0和1。

2. 计算第三位的时候,看第二位b的next值,为1,则把b和1对应的a进行比较,不同,则第三位a的next的值为1,因为一直比到最前一位,都没有发生比较相同的现象。

3. 计算第四位的时候,看第三位a的next值,为1,则把a和1对应的a进行比较,相同,则第四位a的next的值为第三位a的next值加上1。为2。因为是在第三位实现了其next值对应的值与第三位的值相同。

4. 计算第五位的时候,看第四位a的next值,为2,则把a和2对应的b进行比较,不同,则再将b对应的next值1对应的a与第四位的a进行比较,相同,则第五位的next值为第二位b的next值加上1,为2。因为是在第二位实现了其next值对应的值与第四位的值相同。

5. 计算第六位的时候,看第五位b的next值,为2,则把b和2对应的b进行比较,相同,则第六位c的next值为第五位b的next值加上1,为3,因为是在第五位实现了其next值对应的值与第五位相同。

6. 计算第七位的时候,看第六位c的next值,为3,则把c和3对应的a进行比较,不同,则再把第3位a的next值1对应的a与第六位c比较,仍然不同,则第七位的next值为1。

7. 计算第八位的时候,看第七位a的next值,为1,则把a和1对应的a进行比较,相同,则第八位c的next值为第七位a的next值加上1,为2,因为是在第七位和实现了其next值对应的值与第七位相同。


nextval数组的求解方法:

  • 第一位值为1,nextval[1]=0。
  • 从第二位开始,若要求第i位的nextval[i],将next[i]的值对应的位的值i的值进行比较(例如,当i=3时,next[3]=1对应的位数下的字符为‘a’,第i位的值为‘a’),
    • 当前字符的值当前字符下next值所对应位数的值不同,则当前字符的nextval值当前字符对应的next值
    • 当前字符的值当前字符下next值所对应位数的值相同,当前字符的nextval值当前字符对应next值的nextval值

具体计算nextval的过程如下:

位数
模式串abaabcac
next01122312
nextval01021302

1. 第一位的nextval值必定为0,第二位如果与第一位相同则为0,如果不同则为1。

2. 第三位的next值为1,那么将第三位和第一位进行比较,均为a,相同,则第三位的nextval值第一位的nextval值,为0。

3. 第四位的next值为2,那么将第四位和第二位进行比较,不同,则第四位的nextval值为其next值,为2。

4. 第五位的next值为2,那么将第五位和第二位进行比较,相同,则第五位的nextval值为第二位的nextval值,为1。

5. 第六位的next值为3,那么将第六位和第三位进行比较,不同,则第六位的nextval值为其next值,为3。

6. 第七位的next值为1,那么将第七位和第一位进行比较,相同,则第七位的nextval值为0。

7. 第八位的next值为2,那么将第八位和第二位进行比较,不同,则第八位的nextval值为其next值,为2。

获得nextval,代码示例(《数据结构》严蔚敏):

void get_nextval(SString T, int nextval []){
    i = 1;
    nextval[1] = 0;
    j = 0;
    while(i < T[0]){  // T[0]表示串的长度,具体可参考书中P73
        if (j == 0 || T[i] == T[j]){
            ++i;  ++j;
            if (T[i] != T[j]){
                nextval[i] = j
            } else{
                nextval[i] = nextval[j]
            }
        }else{
            j = nextval[j];
        }
    }
}

练练手:

可在“aaaab”内进行验证:

模式串 a a a a b

next值 0 1 2 3 4

nextval值 0 0 0 0 4

  • 98
    点赞
  • 394
    收藏
    觉得还不错? 一键收藏
  • 8
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 8
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值