kmp算法的实现

来源:互联网 发布:淘宝上代接信用卡电话 编辑:程序博客网 时间:2024/05/19 03:23


【代码思想】

 传统匹配思想是,从目标串Target的第一个字符开始扫描,逐一与模式串的对应字符进行匹配,若该组字符匹配,则检测下一组字符,如遇失配,则退回到Target的第二个字符,重复上述步骤,直到整个Pattern在Target中找到匹配,或者已经扫描完整个目标串也没能够完成匹配为止。

 这样的算法理解起来很简单,实现起来也容易,但是其中包含了过多不必要的操作,也就是在目标串中,有些字符是可以直接跳过,不必检测的。

KMP算法通过一个“有用信息”可以知道目标串中下一个字符是否有必要被检测,这个“有用信息”就是用所谓的“前缀函数(一般数据结构书中的next函数)”来存储的。

 这个函数能够反映出现失配情况时,系统应该跳过多少无用字符(也即模式串应该向右滑动多长距离)而进行下一次检测,在上例中,这个距离为4

 总的来讲,KMP算法有2个难点:

  一是这个前缀函数的求法。 

 二是在得到前缀函数之后,怎么运用这个函数所反映的有效信息避免不必要的检测。


#include<stdio.h>#include<string.h>void makeNext(const char P[],int next[])<span class="comment" style="margin: 0px; padding: 0px; border: none; color: rgb(0, 130, 0); font-family: Consolas, 'Courier New', Courier, mono, serif; line-height: 18px; ">//const 不改变源字符串,只读</span><span style="margin: 0px; padding: 0px; border: none; font-family: Consolas, 'Courier New', Courier, mono, serif; line-height: 18px; ">  </span>{    int q,k;    int m = strlen(P);    next[0] = 0;    for (q = 1,k = 0; q < m; ++q)    {        while(k > 0 && P[q] != P[k])            k = next[k-1];        if (P[q] == P[k])        {            k++;        }        next[q] = k;    }}int kmp(const char T[],const char P[],int next[]){    int n,m;    int i,q;    n = strlen(T);    m = strlen(P);    makeNext(P,next);    for (i = 0,q = 0; i < n; ++i)    {        while(q > 0 && P[q] != T[i])            q = next[q-1];        if (P[q] == T[i])        {            q++;        }        if (q == m)        {            printf("模式串匹配开始位置:%d\n",(i-m+2));        }    }}int main(){    int i;    int next[20]={0};    char T[] = "ababxbabaabcdabddsss";    char P[] = "abcdabd";    printf("%s\n",T);    printf("%s\n",P );    kmp(T,P,next);    for (i = 0; i < strlen(P); ++i)    {        printf("%d ",next[i]);    }    printf("\n");    return 0;}</span>

严蔚敏老师的课本,太简洁了,废了不小的劲,继续努力了,下一个哈弗曼编码!


抓狂抓狂抓狂抓狂抓狂怎么回事我写好的代码,粘贴进来都这样子?昨天都删了!

0 0
原创粉丝点击