KMP算法基本思想有许多博客都写到了,写得也十分形象,不懂得可以参考下面的传送门,我就不解释基本思想了。本文主要给出KMP算法及next数组的计算方法(主要是很多网上的代码本人(相信应该是许多人吧)看不懂,直接拿来用也是各种问题),这里的代码封装性和优化都有待考究,但是基本能实现KMP算法

http://www.cnblogs.com/c-cloud/p/3224788.html

  这里提醒各位看官,while循环次数不是常见的固定次数的循环,而是动态根据实际情况将大家固认为的“一趟循环”分解成几次,看代码时留意这点,对各位看官应该有所帮助。

1.KMP的next数组生成方法。

  根据链接中的15来看,可以根据搜索的字符串strKey的长度确定分析的次数(第一次显然是0),第1-len次分析的长度刚好就是左边字符个数(1-len),相应次数的前缀后最的共同元素的最长长度为对应next数组的值。

  同时我们发现规律:根据前缀和后缀的特点,我们可以从每次分析的字符串(每次分析的字符串都不一样)的最后一个字符开始,向前进行字符匹配。

  每次匹配成功一个字符就后缀索引backIndex-1;

  每次匹配字符失败,需要判断是否已经有部分匹配的字符串,若有,应重置后缀索引backIndex=0,相应next数值为0;若无,前缀索引frontIndex+1即可

  每趟大循环,如果遇到已经计算到next的相应数值则进入分析下一组(这里组的概念类似链接的第15点的分析步骤);若遇到该组没有分析完,但是遇到了一些情况:比如说frontIndex =0,或者匹配成功一个字符,则进入大循环,但是是继续分析还没有分析完的这组

 void cal_next(char * strKey, int * next, int len)
{
int calTimes = ;//计算次数 ,第一次肯定为零
int frontIndex=;//负责在前缀里边索引元素
int backIndex = ;//负责在后缀里边索引元素
//清零
for (frontIndex = ; frontIndex < len; frontIndex++)
{
next[frontIndex] = ;
}
frontIndex = calTimes-;//取得可能的最大前缀和后缀的共同元素的最长的长度0-
backIndex = calTimes-;//0-
while (calTimes<=len)
{
printf("calTimes:%d\r\n",calTimes);
printf("frontIdex:%d\r\n", frontIndex);
printf("backIdex:%d\r\n", backIndex);
while (frontIndex > && strKey[backIndex] != strKey[frontIndex ])//最多calTimes次比较
{
printf(" There is a unmatched char\r\n");
if (next[calTimes - ] != )//已经有部分字符匹配,中途出现了不匹配的字符
{
next[calTimes-] = ;
backIndex = calTimes-;//重置后缀索引
break;
}
frontIndex--;
}
if (frontIndex == && strKey[backIndex ] != strKey[])//最差的情况,一次都没有匹配成功
{
next[calTimes-] =;
calTimes++;
frontIndex = calTimes-;
backIndex = calTimes-;
printf(" There is never matched\r\n");
}
else if (frontIndex == && strKey[backIndex ] == strKey[])//最后一个字符匹配成功
{
next[calTimes - ] +=;
calTimes++;
frontIndex = calTimes-;
backIndex = calTimes-;
printf(" the last char matched\r\n");
}
else
{
if (frontIndex > && strKey[backIndex ] == strKey[frontIndex ]) //匹配成功一个
{
next[calTimes - ]++;
backIndex -= ;
frontIndex--;
printf(" a char matched\r\n");
}
}
}
}

2.KMP算法的实现

  这里的实现步骤其实就是算法了步骤了,应该没有什么可讲的,基本的实现思路跟next的生成方法差不多。

 void kmp(char * strText, char *strKey)
{
int *next = new int[strlen(strKey)];
//计算next 数组
cal_next(strKey,next,strlen(strKey));
//查找
int textLenght = strlen(strText);
int keyLenght = strlen(strKey);
int textIndex = ;
int keyIndex = ;
int successNums = ;
while (textIndex < textLenght)
{
while (textIndex <textLenght && strKey[keyIndex] != strText[textIndex])
{
printf(" There is a unmatched char\r\n");
if (keyIndex != )//已经匹配部分字符中途遇到不匹配的字符
{
textIndex -= next[keyIndex-];
printf(" keyIndex %d next %c\r\n", keyIndex, strText[textIndex]);
keyIndex = ;
}else
textIndex++;
}
if (textIndex == textLenght )//一个都没有匹配成功(textLenght-1 时,一定是不予strKey的“相应字符匹配”,所以才会进入textIndex++)
{
if (successNums==)
printf(" Summary: Never Matched\r\n");
else printf(" Summary: %d Matched\r\n",successNums);
}
else if (strKey[keyIndex] == strText[textIndex])//找到匹配的字符
{
keyIndex++;
textIndex++;
if (keyIndex == keyLenght)
{
keyIndex = ;
successNums++;
printf(" Summary: Matched @textIndex=%d\r\n", textIndex - keyLenght);
}
printf(" There is a Matched Char\r\n");
}
}
}

3.main

 int _tmain(int argc, _TCHAR* argv[])
{
char Search[] = {'a','b','\0'};
char Text[] = "abababdhello";
kmp(Text, Search); while ();
return ;
}
 

【字符串匹配】KMP算法和next数组的c/c++实现的更多相关文章

  1. 字符串匹配-BF算法和KMP算法

    声明:图片及内容基于https://www.bilibili.com/video/av95949609 BF算法 原理分析 Brute Force 暴力算法 用来在主串中查找模式串是否存以及出现位置 ...

  2. 字符串匹配KMP算法中Next[]数组和Nextval[]数组求法

    数据结构课本上给了这么一段算法求nextval9[]数组 int get_nextval(SString T,int &nextval[ ]) { //求模式串T的next函数修正值并存入数组 ...

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

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

  4. 字符串匹配-KMP

    节选自 https://www.cnblogs.com/zhangtianq/p/5839909.html 字符串匹配 KMP O(m+n) O原来的暴力算法 当不匹配的时候 尽管之前文本串和模式串已 ...

  5. 字符串匹配--kmp算法原理整理

    kmp算法原理:求出P0···Pi的最大相同前后缀长度k: 字符串匹配是计算机的基本任务之一.举例,字符串"BBC ABCDAB ABCDABCDABDE",里面是否包含另一个字符 ...

  6. 字符串匹配KMP算法的讲解C++

    转自http://blog.csdn.net/starstar1992/article/details/54913261 也可以参考http://blog.csdn.net/liu940204/art ...

  7. zstu.4194: 字符串匹配(kmp入门题&& 心得)

    4194: 字符串匹配 Time Limit: 1 Sec  Memory Limit: 128 MB Submit: 206  Solved: 78 Description 给你两个字符串A,B,请 ...

  8. 字符串匹配——KMP算法

    关于KMP算法的分析,我觉得这两篇博客写的不错: http://www.ruanyifeng.com/blog/2013/05/Knuth–Morris–Pratt_algorithm.html ht ...

  9. 字符串匹配KMP算法的C语言实现

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

随机推荐

  1. [转]undo log与redo log原理分析

    数据库通常借助日志来实现事务,常见的有undo log.redo log,undo/redo log都能保证事务特性,这里主要是原子性和持久性,即事务相关的操作,要么全做,要么不做,并且修改的数据能得 ...

  2. jvm--2.类加载机制

    3.JVM类加载机制 (1)类加载机制 虚拟机把描述类的数据从Class文件,用ClassLoader ,加载到内存,并对数据进行校验.转换解析和初始化,最终形成虚拟机直接使用的java类型, 这就是 ...

  3. ASP.NET上实现

    ASP.NET上实现 fengzhuang.cs: using System;using System.Collections.Generic;using System.Linq;using Syst ...

  4. gulp工具rename

    gulp 对文件批量重命名 gulp-rename重命名 var gulp = require('gulp'); var rename = require("gulp-rename" ...

  5. JDBC值事务

    事务的四大特性: 原子性, 一致性(比如说A给B转账,A转了之后B的账户增加了,两个都完成才叫一致性),隔离性(A给B转账,A给C转账,AB和AC并发是无关的),永久性(转账之后 不可能复原,就是说不 ...

  6. 在_vimrc中 set noexpandtab python 不起效果

    我ctm,今天配置不让tab转为空格,在_vimrc中set noexpandtab 不起效果. set ts=4也不起效果. 但是在命令行中其效果. 我都不知道咋办了. 问人说我有可能使用的不是那个 ...

  7. java动态调用webservice

    cxf方式 public static Object[] invokeRemoteMethod(String url, String method, Object[] parameters) { Ja ...

  8. gulp学习-gulpfile

    安装gulp 假设已经安装了node 和npm (淘宝的cnpm很适合国内使用). 1.首页全局安装gulp. 1 npm install --global gulp 2.其次局部安装gulp.(注: ...

  9. js中的斐波那契数列法

    //斐波那契数列:1,2,3,5,8,13…… //从第3个起的第n个等于前两个之和 //解法1: var n1 = 1,n2 = 2; for(var i=3;i<101;i++){ var ...

  10. 已知当前地理位置经纬度查询几个点中最近的一个地点demo

    1.首先定义一个点与点之间测算距离的方法 2.然后定义找出基本点和集合中最近的一个点的方法 3.取第一条数据即是最近的点的坐标 public class Location { public int i ...