KMP 复习笔记
KMP 学习(复习)笔记
KMP(Knuth-Morris-Pratt)是算法竞赛中常用的字符串匹配算法之一,它可以有效地利用失配信息来使得匹配全过程中不回溯,从而在线性时间内完成匹配。
本文已有前置算法讲解(来自刘毅学长):Here
原理
设模式串 pattern 为 "utqqutnu",目标串 target 为 "utqlwutqqutnu",使用朴素算法进行匹配时("-" 表示匹配成功,"|" 表示在此字符失配):
utqqutlwutqqutnu
------|
utqqutnu
首先,将两串首部对齐,逐个字符匹配,可见在字符 'l' 处失配,按照朴素算法的思想,我们需要把模式串右移一个字符,然后再从模式串首部开始匹配,即:
utqqutlwutqqutnu
|
utqqutnu
这时发现从第一个字符起就不匹配,还要继续右移 ……
但是,似乎有一种更好的策略:我们可以直接把模式串的开头对齐目标串的 "ut" 处,就可以一次跳过几个字符,并且模式串无需回溯:
utqqutlwutqqutnu
--|
utqqutnu
而接下来这次失配后,本来需要将模式串与 't' 对齐,但事实上并不需要,将模式串直接与 'l' 对齐即可。
utqqutlwutqqutnu
|
utqqutnu
KMP 算法就是利用了失配后的部分匹配信息来选择模式串的移动方式,尽可能地避免无用的匹配。
失配信息的利用
通过上述例子我们可以观察到,如果部分匹配的串有对称的前后缀,则我们可以直接将模式串中部分匹配串的前缀与目标串中部分匹配串的后缀对齐,如:
utqqutlwutqqutnu
------|
utqqutnu
例子中的部分匹配串为 "utqqut",有对称的前后缀 "ut",则可以直接将目标串的第二个 "ut" 与模式串的第一个 "ut" 对齐。
再来看这个例子,模式串为 "ttitty",目标串为 "ttittitty"
ttittittypoi
-----|
ttitty
此时的部分匹配串为 "ttitt",它有两个对称的前后缀,分别是 "tt" 和 "t",我们会想,以 "t" 对齐,可以移动更长的距离,事实上呢?
ttittittypoi
-|
ttitty
在模式串第二个 't' 处失配后,继续匹配,最终结果是匹配失败。
然而,如果我们以 "tt" 对齐,则有:
ttittittypoi
------
ttitty
结果是匹配成功。
这个例子告诉我们,当部分匹配串有多个对称前后缀时,需要选择最长的,以保证匹配结果的正确。
失配信息的推导
事实上,KMP 算法利用的失配信息是与目标串无关的,它仅与模式串有关,我们可以用递推的方法在线性的时间内求出模式串的每个可能的部分匹配串(即所有前缀)前缀的失配信息。
我们定义 fail 数组是一个长度等于模式串长度的数组,它的第 i 个成员代表以模式串前 i 个字符作为部分匹配串时,部分匹配串的最长对称前后缀长度。
| 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8
target| u | t | q | q | u | t | n | u
fail | 0 | 0 | 0 | 0 | 1 | 2 | 0 | 1
推导 fail[i] 的方法如下:
- 如果
fail[i - 1]不为 0,且第i个字符与第fail[i - 1] + 1个字符相同,则fail[i]即为fail[i - 1] + 1; - 如果
fail[i - 1]为 0,且第i个字符与首个字符相同,则fail[i] = 1,否则fail[i] = 0; - 难点:如果
fail[i - 1]不为 0,且第i个字符与第fail[i - 1] + 1个字符不同,则继续对比第i个字符与fail[fail[i - 1]] + 1个字符,一直向前找直到匹配或者找到了 0。
如模式串:agctagcagctagct
加粗的 'a' 与最后一个 't' 不匹配,此时向前找找到 "agctagc" 的最后一个 'c' 的对称位置的后一个字符,发现是 't',则找到前后的 "agct" 是一个对称的前后缀。
匹配
有了 fail 数组,匹配就简单多了,只要根据以下三种情况对应处理即可:
- 如果当前字符匹配,则继续匹配下一个字符;
- 如果当前在模式串的首字符处不匹配,则直接将模式串右移一个字符;
- 否则移动模式串,使模式串中部分匹配串的前缀与目标串中部分匹配串的后缀对齐。
完整代码(POJ 3461)
更新于 2021年03月04日
#include <cstdio>
#include <cstring>
const int MAXN = 1000000;
inline int kmp(char *a, char *b) // 在 a 中寻找 b
{
// 求出字符串长度
int na = strlen(a + 1), nb = strlen(b + 1);
static int fail[MAXN + 1];
fail[1] = 0;
for (int i = 2; i <= nb; i++) {
// 取上一位置的 fail 位置之后的字符,判断是否和该位相同
int j = fail[i - 1];
// 不断地向前找 fail 位置,直到找到 0 位置或可以匹配当前字符
while (j != 0 && b[j + 1] != b[i]) j = fail[j];
// 如果能匹配,设置当前位置的 fail 位置
if (b[j + 1] == b[i])
fail[i] = j + 1;
else
fail[i] = 0; // 找不到匹配位置
}
int res = 0; // 匹配次数
for (int i = 1, j = 0; i <= na; i++) {
// 取上一位置的 fail 位置之后的字符,判断是否和要匹配的字符相同
while (j != 0 && b[j + 1] != a[i]) j = fail[j];
// 这一位可以匹配上
if (b[j + 1] == a[i]) j++;
// 匹配成功
if (j == nb) {
res++;
j = fail[j]; // 为了能匹配重叠串
// j = 0 // 如果不允许重叠匹配
}
}
return res;
}
int main() {
int T = 1;
for (scanf("%d", &T); T--;) {
static char a[MAXN + 2], b[MAXN + 2];
// 下标从 1 开始
scanf("%s %s", a + 1, b + 1);
printf("%d\n", kmp(b, a));
}
return 0;
}
KMP 复习笔记的更多相关文章
- Java基础复习笔记系列 九 网络编程
Java基础复习笔记系列之 网络编程 学习资料参考: 1.http://www.icoolxue.com/ 2. 1.网络编程的基础概念. TCP/IP协议:Socket编程:IP地址. 中国和美国之 ...
- Java基础复习笔记系列 八 多线程编程
Java基础复习笔记系列之 多线程编程 参考地址: http://blog.csdn.net/xuweilinjijis/article/details/8878649 今天的故事,让我们从上面这个图 ...
- Java基础复习笔记系列 七 IO操作
Java基础复习笔记系列之 IO操作 我们说的出入,都是站在程序的角度来说的.FileInputStream是读入数据.?????? 1.流是什么东西? 这章的理解的关键是:形象思维.一个管道插入了一 ...
- Java基础复习笔记系列 五 常用类
Java基础复习笔记系列之 常用类 1.String类介绍. 首先看类所属的包:java.lang.String类. 再看它的构造方法: 2. String s1 = “hello”: String ...
- Java基础复习笔记系列 四 数组
Java基础复习笔记系列之 数组 1.数组初步介绍? Java中的数组是引用类型,不可以直接分配在栈上.不同于C(在Java中,除了基础数据类型外,所有的类型都是引用类型.) Java中的数组在申明时 ...
- Java基础复习笔记基本排序算法
Java基础复习笔记基本排序算法 1. 排序 排序是一个历来都是很多算法家热衷的领域,到现在还有很多数学家兼计算机专家还在研究.而排序是计算机程序开发中常用的一种操作.为何需要排序呢.我们在所有的系统 ...
- Angular复习笔记7-路由(下)
Angular复习笔记7-路由(下) 这是angular路由的第二篇,也是最后一篇.继续上一章的内容 路由跳转 Web应用中的页面跳转,指的是应用响应某个事件,从一个页面跳转到另一个页面的行为.对于使 ...
- Angular复习笔记7-路由(上)
Angular复习笔记7-路由(上) 关于Angular路由的部分将分为上下两篇来介绍.这是第一篇. 概述 路由所要解决的核心问题是通过建立URL和页面的对应关系,使得不同的页面可以用不同的URL来表 ...
- Angular复习笔记6-依赖注入
Angular复习笔记6-依赖注入 依赖注入(DependencyInjection)是Angular实现重要功能的一种设计模式.一个大型应用的开发通常会涉及很多组件和服务,这些组件和服务之间有着错综 ...
- Angular复习笔记5-指令
Angular复习笔记5-指令 在Angular中,指令是一个重要的概念,它作用在特定的DOM元素上,可以扩展这个元素的功能,为元素增加新的行为.本质上,组件可以被理解为一种带有视图的指令.组件继承自 ...
随机推荐
- hive报错Execution Error, return code 2 from org.apache.hadoop.hive.ql.exec.mr.MapRedTask[已解决]
我的报错信息 Execution Error, return code 2 from org.apache.hadoop.hive.ql.exec.mr.MapRedTask 解决1(可行):不走ya ...
- xv6:labs2 syscall
lab2 1.lab2的内容总结:关于系统调用整个跟踪过程: 使用系统调用时,用户态会通过软中断(trap,陷阱)进入内核中,由trap识别中断来自系统调用,然后调用syscall函数, 跟踪过程: ...
- 简易的git命令行入门教程
一.Git 全局设置 git config --global user.name "用户名" git config --global user.email "邮件地址@1 ...
- Tainted canvases may not be exported,视频帧截图跨域
做原生相机拍照的时候遇见的有趣问题,视频流是上传到云服务器的在线链接,赋值到video的src上,然后使用canvas的drawImg方法去截取视频帧做照片,结果canvas报错视频跨域. 解决方案: ...
- 09 - Shell流程控制语句
1. if-else语句 能够使用if条件语句进行条件判断 1.1 if 语法 if 条件 then 命令 fi if 条件; then 命令; fi 1.2 if-else 语法 if 条件 the ...
- Redis本地安装以及使用(详细教程)
Redis 安装 Windows 下载安装 Redis默认端口:6379 整个过程如下: 1.下载连接 https://github.com/tporadowski/redis/releases Re ...
- 《深入理解 FFmpeg》第一章彩色插图汇总
layout: post title: "<深入理解 FFmpeg>第一章彩色插图" tags: - "FFmpeg" 这是<深入理解 FFm ...
- 简单几行实现sliver上线提醒
准备魔改sliver去掉一些特征什么的,这里记录一下最简单实现上线消息通过企业微信机器人提醒的方式,这很简单也有很多不足还需要接着改的 protobuf中对消息Beacon和Session的定义如下, ...
- VSCode C++开发环境配置: LLVM clang clangd
工欲善其事,必先利其器 llvm/clang 比 VSCode 自带的代码提示功能速度更快,功能更强(支持 clang-tidy). 安装 llvm.clang sudo apt install ll ...
- Programming abstractions in C阅读笔记:p184-p195
<Programming Abstractions In C>学习第61天,p184-p195总结. 一.技术总结 1.mutual recursion 2.natural number ...