KMP算法是BF算法的改进,主要是消除了主串指针的回溯,提高算法效率。
先简单介绍一下BF算法:
基本思路:
从目标串s的第一个字符开始和模式串的第一个字符比较,相等逐个比较后续字符,否则从目标串的第二个字符开始从新与模式串的第一个字符进行比较。以此类推。。
其时间复杂度为O(m*n)
BF-CODE:
1 int BF(char s[],char t[]) 2 { 3 int i=0,j=0; 4 int lens=strlen(s); 5 int lent=strlen(t); 6 while(i<lens&&j<lent) 7 { 8 if(s[i]==t[j])//继续匹配下一个字符 9 { 10 i++; 11 j++; 12 }//主串和子串依次匹配下一个字符 13 else 14 { 15 i=i-j+1;//主串从下一个位置开始匹配 16 j=0;//子串从头开始匹配 17 } 18 } 19 if(j>=lent) 20 return (i-lent); 21 else 22 return -1; 23 }
KMP算法:
如何消除主串指针的回溯呢?需要分析模式串t,对于t的每个字符t[j],若存在一个整数K,使得模式t中k所指字符之前的k个字符依次与tj的前面k个字符相同,并与主串s中i所指的字符之前的k个字符相等。那么就可以利用这种信息避免不必要的回溯了。这种信息我们把它记录在模式串的next数组中,那么如何来求next数组呢?
怎么求串的模式值next[n]
GET-NEXT CODE:
1 void get-next(char s[],char t[]) 2 { 3 int j,k; 4 j=0; 5 k=-1; 6 next[0]=-1; 7 int lent=strlen(t); 8 int lens=strlen(s); 9 while(j<lent-1) 10 { 11 if(k==-1||t[j]==t[k]) 12 { 13 j++; 14 k++; 15 next[j]=k; 16 } 17 else 18 k=next[k]; 19 } 20 }
kmp算法的思想是:设s为目标穿,t为模式串,并设i,j指针分别指示目标串和模式串中正待比较的字符,令i,j的初始值均为0 ,若有s[i]=t[j],则i,j分别增加1,否则,j再退回到j=next[j]的位置,继续比较。直到出现下面两种情况:1.j退回到某个j=next[j]的位置时有s[i]=t[j],则指针各增加一后继续匹配,2.j退回到j=-1,令指针各增加1,下一次比较s[i+1]和t[0].时间复杂度是O(N+M).
KMP-CODE:
1 int kmp(char s[],char t[]) 2 { 3 int lent=strlen(t); 4 int lens=strlen(s); 5 int i=0,j=0; 6 while(i<lens&&j<lent) 7 { 8 if(j==-1||s[i]==t[j]) 9 { 10 i++; 11 j++; 12 } 13 else 14 j=next[j];//i不变j退后 15 } 16 if(j>=lent) 17 return (i-lent);//返回匹配模式串的首字符下标 18 else 19 return -1; 20 }
kmp算法基本讲完了,下面来说说Kmp的几个应用:
1.求最小循环节和最大重复次数(此段非原创,摘自http://www.cnblogs.com/jackge/archive/2013/01/05/2846006.html)
在KMP算法的使用中,首要任务就是获取一个字符串的next数组,所以我们得明白next数组的含义(最好的方法是自己弄个例子,在草稿纸上模拟一下),在这里,通俗一点 讲,next[k] 表示,在模式串的 k 个字符失配了,然后下一次匹配从 next[k] 开始(next[k] 中保存的是该失配字符的前一个字符在前面出现过的最近一次失配的字符后面的 一个字符的位置,有点绕口,自己写个例子看看就明白了,也可以继续往下看,有介绍,然后再自己尝试写写 )。
至于next数组为什么可以用来求重复前缀呢,而且求出来的重复前缀是最小的呢?
我们来看一下求next数组的代码:
void getnext(int len){ int i=0,j=-1; next[0]=-1; while(i<len){ if(j==-1 || str[i]==str[j]){ i++;j++; next[i]=j; }else j=next[j]; } }
个人认为,next数组在求解的过程中,用到了KMP的思想,当前失配了,就回溯到上一个next,请见 j=next[j] ,先说个结论,如果到位置 i ,如果有 i%(i- next(i))==0 , 那说明字符串开始循环了,并且循环到 i-1 结束,为什么这样呢?
我们先假设到达位置 i-1 的时候,字符串循环了(到i-1完毕),那么如果到第i个字符的时候,失配了,根据next数组的求法,我们是不是得回溯?
然而回溯的话,由于字符串是循环的了(这个是假定的),next[i] 是不是指向上一个循环节的后面一个字符呢??
是的,上一个循环节的末尾是 next[i]-1 ,然后现在循环节的末尾是 i-1 ,然么循环节的长度是多少呢?
所以,我们有 (i - 1) - ( next[i] - 1 ) = i - next[i] 就是循环节的长度(假设循环成立的条件下),但是我们怎么知道这个循环到底成立吗?
现在我们已经假设了 0————i-1 循环了,那么我们就一共有i 个字符了,如果有 i % ( i - next[i] ) == 0,总的字符数刚好是循环节的倍数,那么说明这个循环是成立的。
注意还有一点,如果 next[i] == 0,即使符合上述等式,这也不是循环的,举个反例
0 1 2 3 4 5
a b c a b d
-1 0 0 0 1 2
下标为1,2,3的next值均为0,那么 i%(i-next【i】)=i%i==0,但是这个并不是循环。
解释完毕,然后再来看下,为什么求出来的循环节长度是最小的呢?
因为next数组失配的时候,总是回溯到最近的循环节,所以i-next【i】就是最小的循环节长度
为什么求出来的循环次数是最多的呢?
循环节长度是最小的了,那么循环次数肯定是最多的了。
总结一下,如果对于next数组中的 i, 符合 i % ( i - next[i] ) == 0 && next[i] != 0 , 则说明字符串循环,而且
循环节长度为: i - next[i]
循环次数为: i / ( i - next[i] )
There is a meaning for wings that cannot fly,it's a previous memory of when you once flew through the sky.这篇博文写得挺认真的,希望大家多多指点~噶呜~~