子串在主串中的定位操作通常称做串的模式匹配。

KMP模式匹配算法实现:
/* Index_KMP.h头文件 */
#include<string>
#include<sstream> void get_next(std::string T,int *next)
{
unsigned int i,j;
i=1;
j=0;
next[1]=0;
while(i<(T.size()-1)) /* 此处T的首个字符T[0]表示串T的长度,不参与计算 */
{
if(j==0||T[i]==T[j]) /* T[i]表示后缀的单个字符,T[j]表示前缀的单个字符 */
{
++i;
++j;
next[i]=j;
}
else
j=next[j]; /* 若字符不相同,则j值回溯 */
}
} int Index_KMP(std::string S, std::string T, unsigned int pos)
{
std::string s,t; /*在字符串S,T的最前插入一个字符来保存串的长度值,*/
std::ostringstream s1,s2; /*用来保证字符串的有用元素是从下标1开始*/
s1<<S.size()<<S; /*将新的数组保存为s,t*/
s2<<T.size()<<T;
s=s1.str();
t=s2.str();
unsigned int i=pos; /* i用于主串s当前位置下标值,若pos不为1,则从pos位置开始匹配 */
unsigned int j=1; /* j用于字串t中当前位置下标值 */
int next[255]; /* 定义一next数组 */
get_next(t,next); /* 对串t作分析,得到next数组 */
while (i<=(s.size()-1) && j<=(t.size()-1)) /* 若i小于S的长度且j小于T的长度时,循环继续 */
{
if ( j==0 || s[i]==t[j]) /* 两字母相等则继续,相对于朴素算法增加了j=0判断 */
{
++i;
++j;
}
else /* 指针后退重新开始匹配 */
{
j=next[j]; /* j退回合适的位置,i值不变 */
}
}
if ( j>(t.size()-1) )
return i-(t.size()-1);
else
return 0;
}

KMP模式匹配算法的改进:


/* Index_KMP.h头文件 */
#include<string>
#include<sstream> void get_nextval(std::string T,int *nextval)
{
unsigned int i,j;
i=1;
j=0;
nextval[1]=0;
while(i<(T.size()-1)) /* 此处T的首个字符T[0]表示串T的长度,不参与计算 */
{
if(j==0||T[i]==T[j]) /* T[i]表示后缀的单个字符,T[j]表示前缀的单个字符 */
{
++i;
++j;
if (T[i]!=T[j]) /*若当前字符与前缀字符不同*/
nextval[i]=j; /*则当前的j为nextval在i位置的值*/
else
nextval[i]=nextval[j]; /*如果与前缀字符相同,则将前缀字符的nextval值赋值给nextval在i位置的值*/ }
else
j=nextval[j]; /* 若字符不相同,则j值回溯 */
}
} int Index_KMP(std::string S, std::string T, unsigned int pos)
{
std::string s,t; /*在字符串S,T的最前插入一个字符来保存串的长度值,*/
std::ostringstream s1,s2; /*用来保证字符串的有用元素是从下标1开始*/
s1<<S.size()<<S; /*将新的数组保存为s,t*/
s2<<T.size()<<T;
s=s1.str();
t=s2.str();
unsigned int i=pos; /* i用于主串s当前位置下标值,若pos不为1,则从pos位置开始匹配 */
unsigned int j=1; /* j用于字串t中当前位置下标值 */
int next[255]; /* 定义一next数组 */
get_nextval(t,next);
while (i<=(s.size()-1) && j<=(t.size()-1)) /* 若i小于S的长度且j小于T的长度时,循环继续 */
{
if ( j==0 || s[i]==t[j]) /* 两字母相等则继续,相对于朴素算法增加了j=0判断 */
{
++i;
++j;
}
else /* 指针后退重新开始匹配 */
{
j=next[j]; /* j退回合适的位置,i值不变 */
}
}
if ( j>(t.size()-1) )
return i-(t.size()-1);
else
return 0;
}

匹配算法不做变化,只需要将"get_next(T,next)"改为“get_nextval (T,next)”即可。



总结:改进过的KMP算法,它是在计算出 next 值的同时,如果a位字符与它 next 值指向的 b 位字符相等,则该 a 位的nextval 就指向 b 位的 nextval 值,如果不等,则该 a 位的 nextval 值就是它自己 a 位的 nextval 的值。

C++编程练习(7)----“KMP模式匹配算法“字符串匹配的更多相关文章

  1. 线性表-串:KMP模式匹配算法

    一.简单模式匹配算法(略,逐字符比较即可) 二.KMP模式匹配算法 next数组:j为字符序号,从1开始. (1)当j=1时,next=0: (2)当存在前缀=后缀情况,next=相同字符数+1: ( ...

  2. 详细解读KMP模式匹配算法

    转载请注明出处:http://blog.csdn.net/fightlei/article/details/52712461 首先我们需要了解什么是模式匹配? 子串定位运算又称为模式匹配(Patter ...

  3. [从今天开始修炼数据结构]串、KMP模式匹配算法

    [从今天开始修炼数据结构]基本概念 [从今天开始修炼数据结构]线性表及其实现以及实现有Itertor的ArrayList和LinkedList [从今天开始修炼数据结构]栈.斐波那契数列.逆波兰四则运 ...

  4. 串、KMP模式匹配算法

    串是由0个或者多个字符组成的有限序列,又名叫字符串. 串的比较: 串的比较是通过组成串的字符之间的编码来进行的,而字符的编码指的是字符在对应字符集中的序号. 计算机中常用的ASCII编码,由8位二进制 ...

  5. KMP模式匹配算法

    KMP模式匹配算法 相信很多人对于这个还有点不了解,或者说是不懂,下面,通过一道题,来解决软考中的这个问题! 正题: aaabaaa,其next函数值为多少? 对于这个问题,我们应该怎么做呢? 1.整 ...

  6. 字符串的模式匹配算法——KMP模式匹配算法

    朴素的模式匹配算法(C++) 朴素的模式匹配算法,暴力,容易理解 #include<iostream> using namespace std; int main() { string m ...

  7. 数据结构(三)串---KMP模式匹配算法

    (一)定义 由于BF模式匹配算法的低效(有太多不必要的回溯和匹配),于是某三个前辈发表了一个模式匹配算法,可以大大避免重复遍历的情况,称之为克努特-莫里斯-普拉特算法,简称KMP算法 (二)KMP算法 ...

  8. 浅谈KMP模式匹配算法

    普通的模式匹配算法(BF算法) 子串的定位操作通常称为模式匹配算法 假设有一个需求,需要我们从串"a b a b c a b c a c b a b"中,寻找内容为"a ...

  9. 初探—KMP模式匹配算法

    KMP算法思想: 普通的字符串匹配算法S主串必须要回溯.但回溯就影响了效率. 改进的地方也就是这里,我们从P 串本身出发,事先就找准了T自身前后部分匹配的位置,那就可以改进算法. next数组的含义: ...

随机推荐

  1. CodeForces 617D Polyline

    无脑暴力判断. #include<cstdio> #include<cstring> #include<vector> #include<cmath> ...

  2. [Unity Asset]AssetBundle系列——游戏资源打包

    转载:http://www.cnblogs.com/sifenkesi/p/3557231.html 将本地资源打包,然后放到资源服务器上供游戏客户端下载或更新.服务器上包含以下资源列表:(1)游戏内 ...

  3. iOS开发 改变UINavigationController的UINavigationBar的高度和背景图片

    1.改变高度 自定义UINavigationBar的新类别: //UINavigationBar+BackgoundImage.h #import <Foundation/Foundation. ...

  4. ZOJ 3927 Programming Ability Test

    水题,判断一下加起来是否大于等于80 #include<cstdio> #include<cstring> #include<cmath> #include< ...

  5. 严重: Servlet.service() for servlet jsp threw exception java.lang.IllegalStateException: getOutputStream() has already been called for this response

    严重: Servlet.service() for servlet jsp threw exception    java.lang.IllegalStateException: getOutputS ...

  6. Extjs4新特性

    Extjs 4相对于之前的版本作出了重大的修正.其中包括全新的类系统.新平台的引入.API的修整和加强还有新组件的引入(如新的图表和图形组件).Extjs 4提供更快速.更稳定的用户体验,并且让开发人 ...

  7. IT技术网站汇总

    首先是比较著名的博客型的网站!一般来说在国外比较著名的博客基本上都是比较有影响力发起的或者建立的经常发布一些比较有思考力深入分析的文章! 博客媒体网站 1.www.ArsTechnica.com 2. ...

  8. 分析$.isPlainObject

    作者:zccst 本次学习$.isPlainObject,是不是一个普通对象.测试对象是否是纯粹的对象(通过 "{}" 或者 "new Object" 创建的) ...

  9. USACO 2015 December Contest, Platinum Problem Max Flow【树链剖分】

    题意比较难理解,就是给你n个点的树,然后给你m个修改操作,每一次修改包括一个点对(x, y),意味着将x到y所有的点权值加一,最后问你整个树上的点权最大是多少. 比较裸的树链剖分了,感谢Haild的讲 ...

  10. redhat在线安装chrome浏览器

    开始的时候是参考吹尽黄沙始到金的文章http://www.cnblogs.com/effective/archive/2012/03/18/2405189.html 1.创建一个文件/etc/yum. ...