KMP算法

一、传统字符串匹配算法

  1. /*
  2. * 从s中第sIndex位置开始匹配p
  3. * 若匹配成功,返回s中模式串p的起始index
  4. * 若匹配失败,返回-1
  5. */
  6. int index(const std::string &s, const std::string &p, const int sIndex = )
  7. {
  8. int i = sIndex, j = ;
  9.  
  10. if (s.length() < || p.length() < || sIndex < )
  11. {
  12. return -;
  13. }
  14.  
  15. while (i != s.length() && j != p.length())
  16. {
  17. if (s[i] == p[j])
  18. {
  19. ++i;
  20. ++j;
  21. }
  22. else
  23. {
  24. i = i - j + ;
  25. j = ;
  26. }
  27. }
  28. return j == p.length() ? i - j: -;
  29. }

另外一种简单匹配:

  1. int Index_BF ( char S [ ], char T [ ], int pos )
  2. {
  3.   /* 若串 S 中从第pos(S 的下标0≤pos<StrLength(S))个字符
  4.   起存在和串 T 相同的子串,则称匹配成功,返回第一个
  5.   这样的子串在串 S 中的下标,否则返回 -1 */
  6.  
  7.   int i = pos, j = ;
  8.   while ( S[i+j] != '\0'&& T[j] != '\0')
  9.   {
        if ( S[i+j] == T[j] )
  10.       j++; // 继续比较后一字符
  11.     else
  12.     {
  13.       i ++;
          j = ; // 重新开始新的一轮匹配
  14.     }  
      }
  15.  
  16.   if ( T[j] == '\0')
  17.      return i; // 匹配成功 返回下标
  18.   else
  19.      return -; // 串S中(第pos个字符起)不存在和串T相同的子串
  20.  
  21. } // Index_BF

二、KMP算法

//http://www.cppblog.com/oosky/archive/2006/07/06/9486.html

定义

(1)next[0]= -1  意义:任何串的第一个字符的模式值规定为-1。

(2)next[j]= -1   意义:模式串T中下标为j的字符,如果与首字符相同,且j的前面的1—k个字符与开头的1—k个字符不等(或者相等但T[k]==T[j])(1≤k<j)。

   如:T=”abCabCad” 则 next[6]=-1,因T[3]=T[6]

(3)next[j]=k    意义:模式串T中下标为j的字符,如果j的前面k个字符与开头的k个字符相等,且T[j] != T[k] (1≤k<j)。

即T[0]T[1]T[2]。。。T[k-1]==T[j-k]T[j-k+1]T[j-k+2]…T[j-1]  且T[j] != T[k].(1≤k<j);

( 4 ) next[j]=0   意义:除(1)(2)(3)的其他情况。

举例

01)求T=“abcac”的模式函数的值。

next[0]= -1  根据(1)

next[1]=0   根据 (4)   因(3)有1<=k<j;不能说,j=1,T[j-1]==T[0]

next[2]=0   根据 (4)   因(3)有1<=k<j;(T[0]=a)!=(T[1]=b)

next[3]= -1  根据 (2)

next[4]=1   根据 (3)  T[0]=T[3] 且 T[1]=T[4]

下标

0

1

2

3

4

T

a

b

c

a

c

next

-1

0

0

-1

1

若T=“abcab”将是这样:

下标

0

1

2

3

4

T

a

b

c

a

b

next

-1

0

0

-1

0

为什么T[0]==T[3],还会有next[4]=0呢, 因为T[1]==T[4], 根据 (3)” 且T[j] != T[k]”被划入(4)。

02)来个复杂点的,求T=”ababcaabc” 的模式函数的值。

  next[0]= -1    根据(1)

next[1]=0    根据(4)

next[2]=-1   根据 (2)

  next[3]=0   根据 (3) 虽T[0]=T[2] 但T[1]=T[3] 被划入(4)

  next[4]=2   根据 (3) T[0]T[1]=T[2]T[3] 且T[2] !=T[4]

  next[5]=-1  根据 (2)

  next[6]=1   根据 (3) T[0]=T[5] 且T[1]!=T[6]

  next[7]=0   根据 (3) 虽T[0]=T[6] 但T[1]=T[7] 被划入(4)

  next[8]=2   根据 (3) T[0]T[1]=T[6]T[7] 且T[2] !=T[8]

下标

0

1

2

3

4

5

6

7

8

T

a

b

a

b

c

a

a

b

c

next

-1

0

-1

0

2

-1

1

0

2

只要理解了next[3]=0,而不是=1,next[6]=1,而不是= -1,next[8]=2,而不是= 0,其他的好象都容易理解。

03)   来个特殊的,求 T=”abCabCad” 的模式函数的值。

下标

0

1

2

3

4

5

6

7

T

a

b

C

a

b

C

a

d

next

-1

0

0

-1

0

0

-1

4

next[5]= 0  根据 (3) 虽T[0]T[1]=T[3]T[4],但T[2]==T[5]

next[6]= -1  根据 (2) 虽前面有abC=abC,但T[3]==T[6]

next[7]=4   根据 (3) 前面有abCa=abCa,且 T[4]!=T[7]

若T[4]==T[7],即T=” adCadCad”,那么将是这样:next[7]=0, 而不是= 4,因为T[4]==T[7].

下标

0

1

2

3

4

5

6

7

T

a

d

C

a

d

C

a

d

next

-1

0

0

-1

0

0

-1

0

代码

  1. #include <iostream.h>
  2. #include <string.h>
  3. using namespace std;
  4.  
  5. void getNext(char *p,int *next)
  6. {
  7. int j,k;
  8. next[]=-;
  9. j=;
  10. k=-;
  11. while(j<strlen(p)-)
  12. {
  13. if(k==-||p[j]==p[k]) //匹配的情况下,p[j]==p[k]
  14. {
  15. j++;
  16. k++;
  17. next[j]=k;
  18. }
  19. else //p[j]!=p[k]
  20. k=next[k];
  21. }
  22. }
  23.  
  24. int KMPMatch(char *s,char *p)
  25. {
  26. int next[];
  27. int i,j;
  28. i=;
  29. j=;
  30. getNext(p,next);
  31. while(i<strlen(s))
  32. {
  33. if(j==-||s[i]==p[j])
  34. {
  35. i++;
  36. j++;
  37. }
  38. else
  39. {
  40. j=next[j]; //消除了指针i的回溯
  41. }
  42. if(j==strlen(p))
  43. return i-strlen(p);
  44. }
  45. return -;
  46. }
  47.  
  48. int main()//abCabCad
  49. {
  50. char* text="bababCabCadcaabcaababcbaaaabaaacababcaabc";
  51. char* pattern="abCab";
  52.  
  53. cout<< KMPMatch(text, pattern) << endl;
  54.  
  55. return ;
  56. }

Algorithm --> KMP算法的更多相关文章

  1. 【★】KMP算法完整教程

    KMP算法完整教程 全称:                               Knuth_Morris_Pratt Algorithm(KMP算法) 类型:                 ...

  2. 【★】KMP算法完整教程

    KMP算法完整教程 全称:                               Knuth_Morris_Pratt Algorithm(KMP算法) 类型:                 ...

  3. KMP算法完整教程 (上)

    KMP算法完整教程 全称: Knuth_Morris_Pratt Algorithm(KMP算法) 类型: 高级检索算法 功能: 字符串匹配查找 提出者: D.E.Knuth(克努兹),J.H.Mor ...

  4. !KMP算法完整教程

      KMP算法完整教程 全称:                               Knuth_Morris_Pratt Algorithm(KMP算法) 类型:                ...

  5. 从有限状态机的角度去理解Knuth-Morris-Pratt Algorithm(又叫KMP算法)

    转载请加上:http://www.cnblogs.com/courtier/p/4273193.html 在开始讲这个文章前的唠叨话: 1:首先,在阅读此篇文章之前,你至少要了解过,什么是有限状态机, ...

  6. 笔试算法题(52):简介 - KMP算法(D.E. Knuth, J.H. Morris, V.R. Pratt Algorithm)

    议题:KMP算法(D.E. Knuth, J.H. Morris, V.R. Pratt Algorithm) 分析: KMP算法用于在一个主串中找出特定的字符或者模式串.现在假设主串为长度n的数组T ...

  7. [Algorithm] 字符串匹配算法——KMP算法

    1 字符串匹配 字符串匹配是计算机的基本任务之一. 字符串匹配是什么?举例来说,有一个字符串"BBC ABCDAB ABCDABCDABDE",我想知道,里面是否包含另一个字符串& ...

  8. 萌新笔记——用KMP算法与Trie字典树实现屏蔽敏感词(UTF-8编码)

    前几天写好了字典,又刚好重温了KMP算法,恰逢遇到朋友吐槽最近被和谐的词越来越多了,于是突发奇想,想要自己实现一下敏感词屏蔽. 基本敏感词的屏蔽说起来很简单,只要把字符串中的敏感词替换成"* ...

  9. 【模式匹配】KMP算法的来龙去脉

    1. 引言 字符串匹配是极为常见的一种模式匹配.简单地说,就是判断主串\(T\)中是否出现该模式串\(P\),即\(P\)为\(T\)的子串.特别地,定义主串为\(T[0 \dots n-1]\),模 ...

随机推荐

  1. VS2005 添加onTimer定时器

    SetTimer(1,300,NULL); void CchangeDisplayDlg::OnTimer(UINT_PTR nIDEvent) { // TODO: 在此添加消息处理程序代码和/或调 ...

  2. 通过grub-install命令把grub安装到u盘-总结

    通过grub-install命令把grub安装到u盘 ①准备一个u盘,容量不限,能有1MB都足够了. ②把u盘格式化(我把u盘格式化成FAT.fat32格式了,最后证明也是成功的).③开启linux系 ...

  3. Java中的“&”和“&&”的区别

    Java中的"&"和"&&"的区别 1."&"是位运算符,"&&"是逻辑 ...

  4. JavaScript获取select下拉框中的第一个值

    JavaScript获取select下拉框中的第一个值 1.说明 获取select下拉框中的第一个值 2.实现源码 <!DOCTYPE html PUBLIC "-//W3C//DTD ...

  5. 实例 centos自动挂载、备份windows共享文件夹,并删除第7日前当天的备份

    此为用户twk的备份执行方案(192.168.42.246虚拟机数据库,备份到192.168.42.147的第二硬盘)此为用户twk的备份执行方案(192.168.42.5虚拟机数据库,备份到192. ...

  6. 爬虫抓包工具Fiddle设置

    安装证书(用于https)

  7. Codeforces Round #467 (div.2)

    Codeforces Round #467 (div.2) 我才不会打这种比赛呢 (其实本来打算打的) 谁叫它推迟到了\(00:05\) 我爱睡觉 题解 A. Olympiad 翻译 给你若干人的成绩 ...

  8. 【CJOJ2316】【模板】可持久化线段树

    题面 Description 这是一道非常直白的可持久化线段树的练习题,目的并不是虐人,而是指导你入门可持久化数据结构. 线段树有个非常经典的应用是处理RMQ问题,即区间最大/最小值询问问题.现在我们 ...

  9. javascript 推箱子游戏介绍及问题

    最近没什么事情,我的一个亲戚在学校学习PHP,课程中老师让他们编写一个javascript版本的推箱子小游戏,他没什么头绪,就来问我,我当时很闲,就随口答应他包在我身上.结果真正写的时候还是花了点时间 ...

  10. js表单验证处理和childNodes 和children 的区别

    一.对提交表单进行空值验证 html代码: <form action="#"onsubmit="return validate_form(this);" ...