这个问题阮一峰老师讲的很清楚,链接

这里我只贴一下我的C++实现代码:

#include <iostream>
#include <cstring>
#include <string>
#include <set>
#include <map>
using namespace std; void BuildPatchMatchTable(int *partMatchTable, char *findstr)
{
if(findstr == NULL)
return;
partMatchTable[] = ;
int sizefind = strlen(findstr);
for(int i = ; i < sizefind; ++i)
{
set<string> preset;
string tmppre = "";
tmppre = findstr[];
preset.insert(tmppre);
for(int j = ; j < i; ++j)
{
tmppre = tmppre + findstr[j];
preset.insert(tmppre);
} set<string> postset;
string tmppost = "";
tmppost = findstr[i];
postset.insert(tmppost);
for(int j = i - ; j > ; --j)
{
tmppost = findstr[j] + tmppost;
postset.insert(tmppost);
}
set<string> comset;
for(set<string>::iterator beg = preset.begin(); beg != preset.end(); ++beg)
{
if(postset.count(*beg) > )
comset.insert(*beg);
}
int maxlen = ;
for(set<string>::iterator beg = comset.begin(); beg != comset.end(); ++beg)
{
if((*beg).size() > maxlen)
maxlen = (*beg).size();
}
partMatchTable[i] = maxlen;
}
} int kmp(char *srcstr, char *findstr)
{
if(srcstr == NULL || findstr == NULL)
return -;
int lensrc = strlen(srcstr);
int lenfind = strlen(findstr);
int *partMatchTable = new int[lenfind];
BuildPatchMatchTable(partMatchTable, findstr);
for(int i = ; i < lenfind; ++i)
cout << findstr[i] << "\t" << partMatchTable[i] << endl;
int curFind = ;
for(int i = ; i < lensrc; )
{
if(findstr[curFind] == srcstr[i])
{
++i;
++curFind;
}
else
{
if(curFind == )
++i;
else
{
int movestep = curFind - partMatchTable[curFind-];
i += movestep;
curFind = ;
}
}
if(curFind == lenfind)
{
delete []partMatchTable;
return i - lenfind;
}
}
return -;
delete []partMatchTable;
}
int main()
{
char srcStr[] = "bbc abcdab abcdabcdabde";
char findStr[] = "abcdabd";
cout << "pos:" << kmp(srcStr, findStr) << endl; char srcStr2[] = "bbc abcdab abcdabcdabdezzz";
char findStr2[] = "zzz";
cout << "pos:" << kmp(srcStr2, findStr2) << endl; char srcStr3[] = "bbc abcdab abcdabcdabde";
char findStr3[] = "zzz";
cout << "pos:" << kmp(srcStr3, findStr3) << endl;
}

关键问题

1. 求出部分匹配值表

2. 移动次数= 已匹配个数 - 最后一个匹配的字符的部分匹配结果

KMP算法的C++实现的更多相关文章

  1. 简单有效的kmp算法

    以前看过kmp算法,当时接触后总感觉好深奥啊,抱着数据结构的数啃了一中午,最终才大致看懂,后来提起kmp也只剩下“奥,它是做模式匹配的”这点干货.最近有空,翻出来算法导论看看,原来就是这么简单(先不说 ...

  2. KMP算法

    KMP算法是字符串模式匹配当中最经典的算法,原来大二学数据结构的有讲,但是当时只是记住了原理,但不知道代码实现,今天终于是完成了KMP的代码实现.原理KMP的原理其实很简单,给定一个字符串和一个模式串 ...

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

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

  4. KMP算法实现

    链接:http://blog.csdn.net/joylnwang/article/details/6778316 KMP算法是一种很经典的字符串匹配算法,链接中的讲解已经是很明确得了,自己按照其讲解 ...

  5. 数据结构与算法JavaScript (五) 串(经典KMP算法)

    KMP算法和BM算法 KMP是前缀匹配和BM后缀匹配的经典算法,看得出来前缀匹配和后缀匹配的区别就仅仅在于比较的顺序不同 前缀匹配是指:模式串和母串的比较从左到右,模式串的移动也是从 左到右 后缀匹配 ...

  6. 扩展KMP算法

    一 问题定义 给定母串S和子串T,定义n为母串S的长度,m为子串T的长度,suffix[i]为第i个字符开始的母串S的后缀子串,extend[i]为suffix[i]与字串T的最长公共前缀长度.求出所 ...

  7. 字符串模式匹配之KMP算法图解与 next 数组原理和实现方案

    之前说到,朴素的匹配,每趟比较,都要回溯主串的指针,费事.则 KMP 就是对朴素匹配的一种改进.正好复习一下. KMP 算法其改进思想在于: 每当一趟匹配过程中出现字符比较不相等时,不需要回溯主串的 ...

  8. 算法:KMP算法

    算法:KMP排序 算法分析 KMP算法是一种快速的模式匹配算法.KMP是三位大师:D.E.Knuth.J.H.Morris和V.R.Pratt同时发现的,所以取首字母组成KMP. 少部分图片来自孤~影 ...

  9. BF算法与KMP算法

    BF(Brute Force)算法是普通的模式匹配算法,BF算法的思想就是将目标串S的第一个字符与模式串T的第一个字符进行匹配,若相等,则继续比较S的第二个字符和 T的第二个字符:若不相等,则比较S的 ...

  10. KMP算法-next函数求解

    KMP函数求解:一种改进的字符串匹配算法,由D.E.Knuth,J.H.Morris和V.R.Pratt同时发现,因此人们称它为KMP算法.KMP算法的关键是利用匹配失败后的信息,尽量减少模式串与主串 ...

随机推荐

  1. Tmall Programmer Triples Smartisan Sales

    页面程序造假,丢脸丢到华尔街日报去咯 http://blogs.wsj.com/chinarealtime/2014/10/13/tmall-programmer-triples-smartisan- ...

  2. [转]bat中的特殊字符,以及需要在bat中当做字符如何处理

    bat中的特殊字符,以及需要在bat中当做字符如何处理 批处理.Bat 中特殊符号的实际作用,Windows 批处理中特殊符号的作用: @ \\隐藏命令的回显. ~ \\在for中表示使用增强的变量扩 ...

  3. Oracle中的for语句

    for语句是一个可预置循环次数的循环控制语句,他是一个循环计数器,通常是一个整形变量,通过这个循环计数器来控制循环执行的次数 语法如下: for variable_counter_name in [e ...

  4. MVC学习系列——ActionResult扩展

    首先,MVC扩展性非常强. 我从ActionResult扩展入手,因为我们知道微软ActionResult和其子类,有时候并不能满足所有返回值. 比如:我需要返回XML. 因此,现在我扩展XMLRes ...

  5. java & xml parser

    参考: JDK8 API: http://docs.oracle.com/javase/8/docs/api/ DOM: http://www.w3.org/TR/2004/REC-DOM-Level ...

  6. 深入理解ThreadLocal(一)

    Android里,在不同的线程(假设子线程已经创建了Looper)中创建Handler时,并不需要显式指定Looper,系统能自动找到该线程自己的Looper.不同线程的Looper相互独立,之所以能 ...

  7. LintCode-Compare Strings

    Compare two strings A and B, determine whether A contains all of the characters in B. The characters ...

  8. mysql存储过程中传decimal值会自动四舍五入,没有小数

    通过 call  proc(0.2,0.5);  查看结果数据库竟然是0  和 1 原因:proc的参数没有设置好 参数:原本是  in a decimal,in b decimal 应该改为:in ...

  9. 模仿易信的UI

    易信,它的UI还是很简洁,因此本人想模仿一下它,用了一天的时候来研究它的资源文件,终于被我写出来.先看下效果图吧.     (一)首页的标题     main_title.xml <?xml v ...

  10. Jquery 固定悬浮层以及固定表头

    /* =========================================================== * jquery.autofix_anything.js v1 * === ...