引言

一个算是冷门的算法(在竞赛上),不过其算法思想值得深究。

前置知识

  1. kmp的算法思想,具体可以参考 → Click here
  2. trie树(字典树)

正文

问题定义:给定两个字符串 S 和 T(长度分别为 n 和 m),下标从 0 开始,定义 extend[i] 等于 S[i]...S[n-1] 与 T 的最长相同前缀的长度,求出所有的 extend[i]。举个例子,看下表:

i 0 1 2 3 4 5 6 7
S a a a a a b b b
T a a a a a c
extend[i] 5 4 3 2 1 0 0 0

为什么说这是 KMP 算法的扩展呢?显然,如果在 S 的若干个位置 i 有 extend[i] 等于 m,则可知在 S 中找到了匹配串 T,并且匹配的首位置是 i,这就是标准的KMP问题。但是,扩展 KMP 算法可以找到 S 中所有 T 的匹配。接下来具体介绍下这个算法。

算法流程

(1)

如上图,假设当前遍历到 S 串位置 i,即 extend[0]...extend[i - 1]这 i 个位置的值已经计算得到。设置两个变量,a 和 p。p 代表以 a 为起始位置的字符匹配成功的最右边界,也就是 "p = 最后一个匹配成功位置 + 1"。相较于字符串 T 得出,S[a...p) 等于 T[0...p-a)

再定义一个辅助数组 int next[],其中 next[i] 含义为:T[i]...T[m - 1] 与 T 的最长相同前缀长度,m 为串 T 的长度。举个例子:

i 0 1 2 3 4 5
T a a a a a c
next[i] 6 4 3 2 1 0

(2)

S[i] 对应 T[i - a],如果 i + next[i - a] < p,如上图,三个椭圆长度相同,根据 next 数组的定义,此时 extend[i] = next[i - a]

(3)

如果 i + next[i - a] == p 呢?如上图,三个椭圆都是完全相同的,S[p] != T[p - a]T[p - i] != T[p - a],但 S[p] 有可能等于 T[p - i],所以我们可以直接从 S[p]T[p - i] 开始往后匹配,加快了速度。

(4)

如果 i + next[i - a] > p 呢?那说明 S[i...p)T[i-a...p-a) 相同,注意到 S[p] != T[p - a]T[p - i] == T[p - a],也就是说 S[p] != T[p - i],所以就没有继续往下判断的必要了,我们可以直接将 extend[i] 赋值为 p - i

(5)最后,就是求解 next 数组。我们再来看下 next[i]extend[i] 的定义:

  • next[i]T[i]...T[m - 1] 与 T 的最长相同前缀长度;
  • extend[i]S[i]...S[n - 1] 与 T 的最长相同前缀长度。

恍然大悟,求解 next[i] 的过程不就是 T 自己和自己的一个匹配过程嘛,下面直接看代码。

代码

#include<iostream>
#include<algorithm> using namespace std; //求解 T 中 next[],注释参考 GetExtend()
void GetNext(string& T, int& m, int next[]) {
int a = 0, p = 0;
next[0] = m; for (int i = 1; i < m; ++i)
if (i >= p || i + next[i - a] >= p) {
if (i >= p)
p = i; while (p < m && T[p] == T[p - i])
p++; next[i] = p - i;
a = i;
}
else
next[i] = next[i - a];
} // 求解 extend[]
void GetExtend(string& S, int& n, string& T, int& m, int extend[], int next[]) {
int a = 0, p = 0;
GetNext(T, m, next); for (int i = 0; i < n; ++i) {
// i >= p 的作用:举个典型例子,S 和 T 无一字符相同
if (i >= p || i + next[i - a] >= p) {
if (i >= p)
p = i; while (p < n && p - i < m && S[p] == T[p - i])
p++; extend[i] = p - i;
a = i;
}
else
extend[i] = next[i - a];
}
} int main() {
int next[100], extend[100];
string S, T;
int n, m; while (cin >> S >> T) {
int n = S.length();
int m = T.length();
GetExtend(S, n, T, m, extend, next); // 打印 next
cout << "next: ";
for (int i = 0; i < m; ++i)
cout << next[i] << " "; // 打印 extend
cout << "\nextend: ";
for (int i = 0; i < n; i++)
cout << extend[i] << " "; cout << endl << endl;
}
return 0;
}

数据测试如下:

aaaaabbb
aaaaac
next: 6 4 3 2 1 0
extend: 5 4 3 2 1 0 0 0 abc
def
next: 3 0 0
extend: 0 0 0

参考

OI Wiki:https://oi-wiki.org/string/z-func/

拓展kmp算法总结:https://blog.csdn.net/dyx404514/article/details/41831947

神奇的字符串匹配:扩展KMP算法的更多相关文章

  1. Luogu 3375 【模板】KMP字符串匹配(KMP算法)

    Luogu 3375 [模板]KMP字符串匹配(KMP算法) Description 如题,给出两个字符串s1和s2,其中s2为s1的子串,求出s2在s1中所有出现的位置. 为了减少骗分的情况,接下来 ...

  2. 字符串匹配的 KMP算法

    一般字符串匹配过程 KMP算法是字符串匹配算法的一种改进版,一般的字符串匹配算法是:从主串(目标字符串)和模式串(待匹配字符串)的第一个字符开始比较,如果相等则继续匹配下一个字符, 如果不相等则从主串 ...

  3. 字符串匹配的kmp算法 及 python实现

    一:背景 给定一个主串(以 S 代替)和模式串(以 P 代替),要求找出 P 在 S 中出现的位置,此即串的模式匹配问题. Knuth-Morris-Pratt 算法(简称 KMP)是解决这一问题的常 ...

  4. HDU 1711 Number Sequence (字符串匹配,KMP算法)

    HDU 1711 Number Sequence (字符串匹配,KMP算法) Description Given two sequences of numbers : a1, a2, ...... , ...

  5. 字符串匹配(KMP 算法 含代码)

    主要是针对字符串的匹配算法进行解说 有关字符串的基本知识 传统的串匹配法 模式匹配的一种改进算法KMP算法 网上一比較易懂的解说 小样例 1计算next 2计算nextval 代码 有关字符串的基本知 ...

  6. 实现字符串匹配的KMP算法

    KMP算法是Knuth-Morris-Pratt算法的简称,它主要用于解决在一个长字符串S中匹配一个较短字符串s. 首先我们从整体来把我这个算法的思想. 字符串匹配的朴素算法: 我们容易想到朴素算法, ...

  7. 字符串匹配的KMP算法

    ~~~摘录 来源:阮一峰~~~ 字符串匹配是计算机的基本任务之一. 举例来说,有一个字符串”BBC ABCDAB ABCDABCDABDE”,我想知道,里面是否包含另一个字符串”ABCDABD”? 许 ...

  8. 字符串匹配的KMP算法详解及C#实现

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

  9. 字符串匹配与KMP算法实现

    >>字符串匹配问题 字符串匹配问题即在匹配串中寻找模式串是否出现, 首先想到的是使用暴力破解,也就是Brute Force(BF或蛮力搜索) 算法,将匹配串和模式串左对齐,然后从左向右一个 ...

  10. 字符串匹配的KMP算法(转)

    转载:http://kb.cnblogs.com/page/176818/ 字符串匹配是计算机的基本任务之一. 举例来说,有一个字符串"BBC ABCDAB ABCDABCDABDE&quo ...

随机推荐

  1. 23种设计模式 - 对象创建(FactoryMethod - AbstractFactory - Prototype - Builder)

    其他设计模式 23种设计模式(C++) 每一种都有对应理解的相关代码示例 → Git原码 ⌨ 对象创建 通过"对象创建" 模式绕开new,来避免对象创建(new)过程中所导致的紧耦 ...

  2. 平衡二叉搜索树/AVL二叉树 C实现

    //AVTree.h #ifndef MY_AVLTREE_H #define MY_AVLTREE_H typedef int ElementType; struct TreeNode { Elem ...

  3. 项目实战 - 原理讲解<-> Keras框架搭建Mtcnn人脸检测平台

    Mtcnn它是2016年中国科学院深圳研究院提出的用于人脸检测任务的多任务神经网络模型,该模型主要采用了三个级联的网络,采用候选框加分类器的思想,进行快速高效的人脸检测.这三个级联的网络分别是快速生成 ...

  4. Mysql批量导入

    这应该是我写Mysql技术的最后一章了吧,短时间内应该不会再写Mysql的文章了,当然疑难杂症除外 insert语句优化 因为之前我也遇到过这样的问题,是我在做数据库适配的时候碰见的,那是我的数据还是 ...

  5. Solon详解(二)- Solon的核心

    Solon详解系列文章: Solon详解(一)- 快速入门 Solon详解(二)- Solon的核心 Solon详解(三)- Solon的web开发 Solon详解(四)- Solon的事务传播机制 ...

  6. 推荐掌握Linux shell中这7种运算命令

    #常见的算术运算符号 .+.-:加减 .*./.%:乘.除.取余 .**:幂运算 .++.--:增加记减少 .!.&&.||:取反,并且,或 .<,<=,>,=> ...

  7. Linux:安装php

    一.环境准备(lamp环境:linux+apache+php+mysql) 1.安装php之前,首先要检查一下相关lib库 rpm -qa  zlib libxml libjpeg freetype ...

  8. MySQL 8 安装教程(个人版)+创建用户

    Mysql 8的安装教程 解压到指定目录如:D:\WinInstall\mysql-8.0.19-winx64这时候你需要在根目录下创建两个文件,分别是data文件夹和my.ini文件,然后使用编辑器 ...

  9. SpringBoot写后端接口,看这一篇就够了!

    摘要:本文演示如何构建起一个优秀的后端接口体系,体系构建好了自然就有了规范,同时再构建新的后端接口也会十分轻松. 一个后端接口大致分为四个部分组成:接口地址(url).接口请求方式(get.post等 ...

  10. CAS导致的ABA问题以及解决方案

    CAS算法实现一个重要前提需要取出内存中某时刻的数据,而在下时刻比较并替换,那么在这个时间差类会导致数据的变化. 上篇文章讲到CAS会出现一个ABA问题.那什么是ABA问题呢? 官方一点的解释就是:当 ...