KMP算法简单回顾
前言
虽从事企业应用的设计与开发,闲暇之时,还是偶尔涉猎数学和算法的东西,本篇根据个人角度来写一点关于KMP串匹配的东西,一方面向伟人致敬,另一方面也是练练手,头脑风暴。我在自娱自乐,路过的朋友别太认真,嘿
背景
目标串: T(1…..n)
模式串: P(1…..m)
输出:搜索P在T中的位置 s,令 T(s…s+m-1) === P(1…m)
例如:
a g c t a g c a g c t a g c t g中查找a g c t g 返回 12(从1计数)
资料
资料太多了,我在此不准备进行深入分析整个推算过程,所以本篇文章只写给自己看
E文好理解能力好的直接去看原篇论文《Fast Pattern Matching in Strings》
主要思想参考这里:
主要实现参考这里:
http://blog.csdn.net/yearn520/article/details/6729426
http://biaobiaoqi.me/blog/2013/05/25/kmp-algorithm/
笨拙的大脑
我和很多普通人一样,对于此类问题,大脑中的第一反应时朴素暴力匹配法
- 假设T从i处开始迭代索引
- 比较失配时,使用回溯法,将P相对于T右移1位,然后T的迭代索引回溯到i+1,重新匹配
- 缺陷:重复的移动操作,重复的比较匹配操作
- 效率:移动操作为(n-m),比较操作为m,数量级O(n*m)
优化的KMP
KMP的核心思想是,利用模式串进行自我匹配,生成前缀数组,优化移动和匹配操作
- 目标串无须回溯,只需要移动模式串即可,节省移动操作
- 重复的匹配可以省略,节省了匹配操作,大概列两种极端的例子
- 额外的空间开销O(m)(存储nexts数组),线性时间开销 O(n+m)
难点
算法的关键和难点在于计算前缀数组next[]
实现
- 前缀数组next实现
static void SetPrefix(IList<char> pattern, int[] nexts)
{
nexts[0] = 0;
int m = pattern.Count;
for (int i = 1; i < m; i++)
{
int k = nexts[i - 1];
while (pattern[i] != pattern[k] && k != 0)
{
k = nexts[k - 1];
}
if (pattern[i] == pattern[k])
nexts[i] = k + 1;
else
nexts[i] = 0;
}
}
- KMP匹配方法实现
static IEnumerable<int> KMP(string text, string pattern)
{
int[] nexts = new int[pattern.Length]; int n = text.Length;
int m = pattern.Length; SetPrefix(pattern.ToList(), nexts); for (int i = 0, j = 0; i < n; i++)
{
while (j > 0 && text[i] != pattern[j])
j = nexts[j - 1]; if (text[i] == pattern[j])
j++; if (j == m)
yield return i - m + 1;
} yield break;
}
待续
下面列一些字符串模式匹配的其他算法,以后有时间好好研究和实现
- Boyer–Moore string search algorithm
- Boyer-Moore-Horspool string search algorithm
- Apostolico-Giancarlo string search algorithm
- Aho-Corasick multi-pattern string search algorithm
- Rabin-Karp multi-pattern string search algorithm
- Suffix trees
结语
关于串匹配的算法还有很多,关于KMP的改进实现也有很多,时间和精力有限,以后有时间再深入!
KMP算法简单回顾的更多相关文章
- KMP 算法简单解释
讲KMP算法,离不开BF,实际上,KMP就是BF升级版,主要流程和BF一样 不同是在匹配失败时能利用子串的特征减少回溯,利用根据子串特征生成的Next数组来减少 <( ̄︶ ̄)↗[GO!] ...
- 串的应用与kmp算法讲解--学习笔记
串的应用与kmp算法讲解 1. 写作目的 平时学习总结的学习笔记,方便自己理解加深印象.同时希望可以帮到正在学习这方面知识的同学,可以相互学习.新手上路请多关照,如果问题还请不吝赐教. 2. 串的逻辑 ...
- 简单有效的kmp算法
以前看过kmp算法,当时接触后总感觉好深奥啊,抱着数据结构的数啃了一中午,最终才大致看懂,后来提起kmp也只剩下“奥,它是做模式匹配的”这点干货.最近有空,翻出来算法导论看看,原来就是这么简单(先不说 ...
- 简单kmp算法(poj3461)
题目简述: 给你两个字符串p和s,求出p在s中出现的次数. 思路简述: 在介绍看BF算法时,终于了解到了大名鼎鼎的KMP算法,结果属于KMP从入门到放弃系列,后来看了几位大神的博客,似乎有点懂了.此题 ...
- KMP算法实践与简单分析
一.理解next数组 1.约定next[0]=-1,同时可以假想在sub串的最前面有一个通配符"*",能够任意匹配.对应实际的代码t<0时的处理情况. 2.next[j]可以 ...
- 运用kmp算法解决的一些问题的简单题解
学习kmp算法我最后是看的数据结构书上的一本教材学会的..我认为kmp相对于普通的BF算法就是避免了非常多不必要的匹配.而kmp算法的精髓自然就在于next数组的运用...而next数组简而言之就是存 ...
- KMP算法的一个简单实现
今天学习KMP算法,参考网上内容,实现算法,摘录网页内容并记录自己的实现如下: 原文出处: http://www.ruanyifeng.com/blog/2013/05/Knuth%E2%80%93M ...
- KMP算法 C#实现 字符串查找简单实现
KMP算法 的C#实现,初级版本 static void Main(string[] args) { #region 随机字符 StringBuilder sb = new StringBuilder ...
- Linux GCC下strstr的实现以及一个简单的Kmp算法的接口
今天做了一道题,要用判断一个字符串是否是另一个字符串的子串,于是查了一下strstr的实现. 代码如下: char *strstr(const char*s1,const char*s2) { con ...
随机推荐
- Android采访开发——2.通用Android基础笔试题
注意finddreams博客: http://blog.csdn.net/finddreams/article/details/44219231 正值跳槽的热季.整理一下Android面试中最常考的笔 ...
- 《Java并发编程实战》第十三章 显示锁 读书笔记
一.Lock与 ReentrantLock Lock 提供一种无条件的.可轮询的.定时的.可中断的锁获取操作,全部加锁和解锁的方法都是显式的. public interface Lock { void ...
- obj-c编程04:类的继承
这第4篇内容比較少,主要说的是obj-c中的类的继承,须要说明的是我仅仅是写了继承中最简单的形式,假设所有展开来说,那就多了去了!关键是如今肚子里还没装够墨水,没法展开啊! 以下的代码中,我们写了2个 ...
- Javascript学习5 - 函数
原文:Javascript学习5 - 函数 在Javascript中,函数和对象是交织在一起的.有些函数的特性与对象相关联.这一点的内容在第六部分会讨论到. 这一部分主要讨论函数与其它比较熟悉的语言( ...
- hdu 逆袭指数
Problem Description 这依然是关于高富帅小明曾经的故事—— 尽管身处逆境,但小明一直没有放弃努力,除了搬砖,小明还研究过东方的八卦以及西方的星座,一直试图在命理上找到自己能够逆袭 ...
- hdu 3709 数字dp(小思)
http://acm.hdu.edu.cn/showproblem.php?pid=3709 Problem Description A balanced number is a non-negati ...
- linux在构建SVNserver
最近搞了一个云计算server,一些尝试部署server相关的东西.作为用显影剂server.首先要考虑的是建立SVNserver.关于构建过程记录.方便以后. 一.安装svn软件.有些云server ...
- hdu 4932 Miaomiao's Geometry(暴力)
题目链接:hdu 4932 Miaomiao's Geometry 题目大意:在x坐标上又若干个点,如今要用若干条相等长度的线段覆盖这些点,若一个点被一条线段覆盖,则必须在这条线的左端点或者是右端点, ...
- RabbitMQ基本管理(上)
1.1.2 显示RabbitMQ进程 查看RabbitMQ进程信息,输入以下命令: <span style="font-size:18px;"><strong& ...
- Silverlight中的Path
原文:Silverlight中的Path 在Silverlight中Path可能由直线.曲线.或者其他简单的图形对象组成.这篇文章旨在介绍如何使用XAML和C#来创建Path. 废话先行 Path可能 ...