转自:http://blog.csdn.net/ddupd/article/details/19899263

KMP算法详解

KMP算法简介:

KMP算法是一种高效的字符串匹配算法,关于字符串匹配最简单的就是BF算法。BF算法是用两个游标分别指向母串S,模式串T,从开头向后面依次比较字符是否相等,如果相等继续同时向后滑动两个游标,不相等的话,T的游标回溯至开头,S的游标回溯至起初游标的下一位,这种算法原理非常简单,小学生都可以想的到。

KMP算法是在BF算法的基础上加以改进的,它的特点是在遇到字符不匹配时候维持母串T的游标不动,而把模式串向右移动,具体移动到哪一个元素下标,这就是算法的核心思想之处了。

假如母串的i处和模式串的j处不匹配,那么就令k=next(j),表示的意思就是:模式串在j处出现不匹配现象,此时应该将模式串向后一定到下标为k的游标处,在此与之前不匹配的元素进行比较。

Kmp算法的本质:

如图所示:

在下标j处出现不匹配,则k = next(j),表示此时应该把下标k移动到原本j对应的位置处,用T[k]跟s[i]进行对比。如果满足这样的条件,则有T[0],T[1],…T[k-1] = S[i-k],S[i-k+1],…S[i-1]

又因为j之前的字符串跟S都匹配,所以又有T[j-k],T[j-k+1],…T[j-1] = S[i-k],S[i-k+1],…S[i-1].所以得出  T[0],T[1],…T[k-1] = T[j-k],T[j-k+1],…T[j-1]。也就是说图中被标记出来前后两个区域的字符串相等,KMP算法的本质就是找出最大的这样一个k值满足T[0],T[1],…T[k-1] = T[j-k],T[j-k+1],…T[j-1]。

K值的求取方法:

K值的求取用到了数学中的递推的思想,求取K值只跟模式串T自身有关,跟母串S半毛钱关系都没有。先假设已经有 next(j) = k,接下来我们就去求next(j+1)的值。这个要分情况讨论:

如果T[k] = T[j]那么就很容易得到 next(j+1) = k+1 = next(j) + 1;

如果T[k] != T[j],那么此时可以将T[0],T[1],…T[k-1],T[k]看做一个模式串,T[j-k],T[j-k+1],…T[j-1],T[j]看做一个母串,此时模式串在k处出现不匹配现象,那么我们获取next(k)= k1的值,判断T[k1]跟T[j]的值是否相等,如果相等那么next(j+1) = k1+1;如果不相等再往下求新的kn的值,直到T[kn]= T[j],或者遍历到了模式串的开头都不想的话,此时就要把i向后移动一个位置,同时用模式串的开头指向i,或者抽象一点就是把模式串开头的前一位下标(-1)指向i。因为下标(-1)是没有意义的,所以此时等效于下标(0)指向母串的i+1。

算法的实现:

这里一共列出了三个版本的kmp算法,其中第一个是本人根据对算法的理解写的,也是最丑的一个,剩下的两个是改编严蔚敏版的《数据结构与算法》一书中的。

 
 //Algorithms.cpp
#include "Algorithms.h"
#include <vector>
#include <iostream>
using namespace std; Algorithms::Algorithms(void)
{
}
Algorithms::~Algorithms(void)
{
} int Algorithms::kmp1(string strS,string strT)
{
int nSize = strT.size();
vector<int> vecNext(nSize,-);
if (nSize >= )
vecNext[] =;
for (int i=;i<nSize;i++)
{
int k = vecNext[i-];
while (k>= && strT[k] != strT[i-] )
k = vecNext[k];
if(k>= && strT[i-] == strT[k])
vecNext[i] = k + ;
else
vecNext[i] = ;
}
for(int i=;i<nSize;i++)
cout<<"the vector is:"<<i<<": "<<vecNext[i]<<endl; int nLength = strS.size();
int nCount = ;
int nPoss = ;
int nPost = ; while(nPoss < nLength)
{
if ( strS[nPoss] == strT[nPost] )
{
nPoss++;
nPost++;
}
else
{
nPost = vecNext[nPost];
if (nPost == -)
{
nPoss++;
nPost++;
}
} if (nPost == nSize )
{
nCount++;
nPost = ;
}
}
return nCount;
} int Algorithms::kmp2(string strS,string strT)
{
int nSize = strT.size();
vector<int> vecNext(nSize);
int i = ;
vecNext[] = -;
int j = -;
while(i<nSize-)
{
if (j==- || strT[i]==strT[j])
{
i++;
j++;
vecNext[i] = j;
}
else
j = vecNext[j];
}
for(int i=;i<nSize;i++)
cout<<"the vector is:"<<i<<": "<<vecNext[i]<<endl; int nLength = strS.size();
int nCount = ;
int nPoss = ;
int nPost = ; while(nPoss < nLength)
{
if ( strS[nPoss] == strT[nPost] )
{
nPoss++;
nPost++;
}
else
{
nPost = vecNext[nPost];
if (nPost == -)
{
nPoss++;
nPost++;
}
} if (nPost == nSize )
{
nCount++;
nPost = ;
}
}
return nCount;
} int Algorithms::kmp3(string strS,string strT)
{
int nSize = strT.size();
vector<int> vecNext(nSize);
int i = ;
vecNext[] = -;
int j = -;
while(i<nSize-)
{
if (j==- || strT[i]==strT[j])
{
i++;
j++;
if (strT[i] == strT[j])
vecNext[i] =vecNext[j];
else
vecNext[i] = j;
}
else
j = vecNext[j];
}
for(int i=;i<nSize;i++)
cout<<"the vector is:"<<i<<": "<<vecNext[i]<<endl; int nLength = strS.size();
int nCount = ;
int nPoss = ;
int nPost = ; while(nPoss < nLength)
{
if ( strS[nPoss] == strT[nPost] )
{
nPoss++;
nPost++;
}
else
{
nPost = vecNext[nPost];
if (nPost == -)
{
nPoss++;
nPost++;
}
} if (nPost == nSize )
{
nCount++;
nPost = ;
}
}
return nCount;
}
 //main.cpp
#include <iostream>
#include <string>
#include <vector>
#include "Algorithms.h"
using namespace std; void main()
{
string str1,str2;
cout<<"please input str1:"<<endl;
cin>>str1;
cout<<"please input str2:"<<endl;
cin>>str2;
//cout<<"the number of substr in str1 is:"<<Algorithms::kmp1(str1,str2)<<endl;
//cout<<"the number of substr in str1 is:"<<Algorithms::kmp2(str1,str2)<<endl;
cout<<"the number of substr in str1 is:"<<Algorithms::kmp3(str1,str2)<<endl;
system("pause");
}

算法评析:

1:Kmp1

先说下我自己写的吧,代码没有书上的简洁,再说说几个为什么。为什么循环要从i=2开始?

因为要用到int k = vecNext[i-1],以及strT[k] != strT[i-1],如果从i=1开始的话,k的起始值=-1,这样就会出现越界的情况,所以就从i=2开始;另外

next(0)=-1,next(1)=0,这都是毫无疑问的东西,所以可以在前两者已知的情况下,向后求解。

2:Kmp2

Kmp2的巧妙之处在于用了while循环,在需要i变化时才变化,否则已知变换j的值(j表示的是next(i))

3:Kmp3

Kmp3是对kmp2的改进,它考虑到了这样的一种情况:首先在T(i) = T(j)的情况下,如果按照kmp2那么next(i+1)= j+1。但是如果T(i+1)=T(j+1)的话,那么此时就无需在拿T(j+1) 跟母串的比较,因为T(i+1)已经比较过了,并且他们不相等,所以不需要再比较T(j+1),

只需要令:       next(i+1) = next(j+1)。

按照kmp2其实是这样的:  next(i+1) = j+1,j+1 = next(j+1),

所以中间省略的一步就是:next(i+1)= j+1。

总结:

关于kmp网上的讲解很多,但是坑爹的错误程序也不少,在验证一个kmp算法程序是否坑爹,只需要把next数组的内容打印出来,将它和自己口算得到的结果比对。百度百科的C++实现的kmp算法就是错误的,坑爹的。

kmp算法详解的更多相关文章

  1. [转] KMP算法详解

    转载自:http://www.matrix67.com/blog/archives/115 KMP算法详解 如果机房马上要关门了,或者你急着要和MM约会,请直接跳到第六个自然段.    我们这里说的K ...

  2. KMP算法详解(转自中学生OI写的。。ORZ!)

    KMP算法详解 如果机房马上要关门了,或者你急着要和MM约会,请直接跳到第六个自然段. 我们这里说的KMP不是拿来放电影的(虽然我很喜欢这个软件),而是一种算法.KMP算法是拿来处理字符串匹配的.换句 ...

  3. 算法进阶面试题01——KMP算法详解、输出含两次原子串的最短串、判断T1是否包含T2子树、Manacher算法详解、使字符串成为最短回文串

    1.KMP算法详解与应用 子序列:可以连续可以不连续. 子数组/串:要连续 暴力方法:逐个位置比对. KMP:让前面的,指导后面. 概念建设: d的最长前缀与最长后缀的匹配长度为3.(前缀不能到最后一 ...

  4. 数据结构4.3_字符串模式匹配——KMP算法详解

    next数组表示字符串前后缀匹配的最大长度.是KMP算法的精髓所在.可以起到决定模式字符串右移多少长度以达到跳跃式匹配的高效模式. 以下是对next数组的解释: 如何求next数组: 相关链接:按顺序 ...

  5. KMP算法详解&&P3375 【模板】KMP字符串匹配题解

    KMP算法详解: KMP算法是一种改进的字符串匹配算法,由D.E.Knuth,J.H.Morris和V.R.Pratt(雾)提出的. 对于字符串匹配问题(such as 问你在abababb中有多少个 ...

  6. 字符串匹配KMP算法详解

    1. 引言 以前看过很多次KMP算法,一直觉得很有用,但都没有搞明白,一方面是网上很少有比较详细的通俗易懂的讲解,另一方面也怪自己没有沉下心来研究.最近在leetcode上又遇见字符串匹配的题目,以此 ...

  7. KMP算法详解-彻底清楚了(转载+部分原创)

    引言 KMP算法指的是字符串模式匹配算法,问题是:在主串T中找到第一次出现完整子串P时的起始位置.该算法是三位大牛:D.E.Knuth.J.H.Morris和V.R.Pratt同时发现的,以其名字首字 ...

  8. KMP算法详解 --- 彻头彻尾理解KMP算法

    前言 之前对kmp算法虽然了解它的原理,即求出P0···Pi的最大相同前后缀长度k. 但是问题在于如何求出这个最大前后缀长度呢? 我觉得网上很多帖子都说的不是很清楚,总感觉没有把那层纸戳破, 后来翻看 ...

  9. 字符串匹配的KMP算法详解及C#实现

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

随机推荐

  1. C++中虚函数的作用浅析

    虚函数联系到多态,多态联系到继承.所以本文中都是在继承层次上做文章.没了继承,什么都没得谈. 下面是对C++的虚函数这玩意儿的理解. 一, 什么是虚函数(如果不知道虚函数为何物,但有急切的想知道,那你 ...

  2. ssi技术初探

    http://blog.sina.com.cn/s/blog_765941620100wiir.html

  3. 初步揭秘node.js中的事件

    当你学习node.js的时候,Events是一个非常重要的需要理解的事情.非常多的Node对象触发事件,你能在文档API中找到很多例子.但是关于如何写自己的事件和监听,你可能还不太清楚.如果你不了解, ...

  4. 取地址符:&

    例子: <?php $a=10; $b = &$a; echo $b; $b=15; echo $a; //结果:10和15 //当$b = &$a 时,a,b的地址相同,对a, ...

  5. Spring各个jar包的简介

    spring.jar是包含有完整发布的单个jar 包,spring.jar中包含除了spring-mock.jar里所包含的内容外其它所有jar包的内容,因为只有在开发环境下才会用到 spring-m ...

  6. 解决Yum安装依赖问题

    导读 最近在网上看到很多关于安装网络Yum源是报错的求助,本小白也曾遇到过此类问题,后找过度娘没有有效的解决办法.最后,经过几番尝试后终于解决,现在将解决方案共享给大家! Yum源及安装 安装过程在之 ...

  7. Unity3D占用内存太大的解决方法

    原地址:http://www.cnblogs.com/88999660/archive/2013/03/15/2961663.html 最近网友通过网站搜索Unity3D在手机及其他平台下占用内存太大 ...

  8. asp.net动态输出透明gif图片

    要使用asp.net动态输出透明gif图片,也就是用Response.ContentType = "image/GIF". 查了国内几个中文资料都没解决,最后是在一个英文博客上找到 ...

  9. 在Java中>、>>、>>>三者的区别

    Java,是由Sun Microsystems公司于1995年5月推出的Java程序设计语言和Java平台的总称.用Java实现的HotJava浏览器(支持Java applet)显示了Java的魅力 ...

  10. 《ASP.NET1200例》ListView 控件与DataPager控件的结合<一>

    分页     在前一部分开始时介绍的原 HTML 设计中内含分页和排序,所以根据规范完整实现该网格的任务尚未完成.我们先分页,然后再排序. ListView 控件中的分页通过引入另一个新控件 Data ...