Z-algorithm

Algorithm

Task

给定一个文本串 \(S\) 和一个模式串 \(T\),求 \(T\) 对于 \(S\) 的每个后缀子串的公共前缀子串。

Limitations

要求时空复杂度均为线性

Solution

设 \(X\) 是一个字符串,则以下表述中,\(X_u\) 代表 \(X\) 的第 \(u\) 个字符,\(X_{u \sim v}\) 代表 \(X\) 的从 \(u\) 起到 \(v\) 结束的字串。

设 \(n = |S|,~m = |T|\)。

考虑按照长度由大到小扫描 \(S\) 的后缀字串,设当前要求 \(S_{i \sim n}\) 与 \(T\) 的公共前缀子串,则 \(\forall k \in [1, ~i),~S_{k \sim n}\) 的答案都已计算完成。

设先前的计算中,匹配到 \(S\) 最远的一次为第 \(p\) 次,即 \(p + ans_p\) 在所有 \(k + ans_k\) 中最大,设 \(q = p + ans_p\)。显然有 \(p < i\)。

首先不妨设 \(q \geq i\)。\(q < i\) 的情况将在下方说明。

设 \(j = i - p + 1\),不难发现 \(T_j = S_i\),即 \(j\) 是 \(S_i\) 的对应匹配位置。

由于所求是公共前缀字串,因此有

\[S_{p \sim q} = T_{1 \sim ans_p}
\]

引入一组辅助变量,设 \(next_j\) 为 \(T_{j \sim m}\) 与 \(T\) 的最长公共前缀子串长度。

根据定义,有

\[T_{j \sim j + next_j} = T_{1 \sim next_j}
\]

分两种情况讨论。

第一种情况,\(j + next_j < q\),即 \(T_{j \sim j + next_j}\) 是 \(S_{p \sim q}\) 的字串,因此有

\[T_{j \sim j + next_j} = S_{i \sim i + next_j}
\]

又因为 \(T_{j \sim j + next_j} = T_{1 \sim next_j}\)(已证),等量代换得到

\[S_{i \sim i + next_j} = T_{1 \sim next_j}
\]

对于 \(S_i + next_j + 1\),则可以用反证法证明其不等于 \(T_{next_j + 1}\),否则由于 \(T_{j + next_j + 1}\) 依然是 \(S_{p \sim q}\) 的字串,所以 \(T_{j + next_j + 1} = T_{next_j + 1}\),这与 \(next_j\) 是最长前缀公共子串矛盾。

因此,对于这种情况,答案即为 \(next_j\)。

第二种情况,\(j + next_j \geq q\),即 \(T_{j \sim j + next_j}\) 不全是是 \(S_{p \sim q}\) 的字串,因此有

首先可以用与第一种情况相同的证明方式证明 \(S_{i \sim q} = T_{1 \sim q - i + 1}\),即 \(q\) 及以前的字符可以与 \(T\) 完美匹配,而对于 \(q\) 后面的字符,我们暴力将其与 \(T\) 匹配,同时更新 \(p\) 和 \(q\) 的位置即可。

对于 \(q < i\) 的情况,显然 \(q = i - 1\),直接继续暴力进行匹配即可。

考虑时间复杂度:

除掉暴力匹配的环节,剩下的部分显然都是单次 \(O(1)\) 完成,因此这一部分的复杂度是线性的。

考虑每次暴力匹配都会让 \(q\) 右移,所以暴力匹配的次数是线性的,而单次暴力匹配是 \(O(1)\) 的,因此算法的时间复杂度是线性的。

考虑 \(next\) 数组的计算:我们发现这相当于令文本串 \(S = T\),只需要预处理 \(next_1\) 与 \(next_2\),可以发现从 \(next_3\) 起,计算所需要的 \(next\) 值都已经在之前被计算过。

Sample

【P5410】 【模板】扩展 KMP

Description

给定一个文本串 \(S\) 和一个模式串 \(T\),求 \(T\) 对于 \(S\) 的每个后缀子串的公共前缀子串。并输出 \(T\) 的每个后缀字串与 \(T\) 的公共前缀子串长度。

Limitations

字符串长度不超过 \(10^5\)

Solution

板板题。算法在实现上比较吃细节,注意比较大于小于的时候是否应该加等于号。记得对拍

Code

#include <cstdio>
#include <cstring>
#include <algorithm> const int maxn = 100005; int nxt[maxn];
char S[maxn], T[maxn]; void Z_algorithm(const char *const A, const char *const B, const int x, const int y, const bool pt); int main() {
freopen("1.in", "r", stdin);
scanf("%s\n%s", S + 1, T + 1);
int x = strlen(S + 1), y = strlen(T + 1);
nxt[1] = y;
Z_algorithm(T, T, y, y, false);
for (int i = 1; i <= y; ++i) {
qw(nxt[i], i == y ? '\n' : ' ', true);
}
Z_algorithm(S, T, x, y, true);
putchar('\n');
return 0;
} void Z_algorithm(const char *const A, const char *const B, const int x, const int y, const bool pt) {
int p = 0, q = 1;
if (!pt) {
while ((q < x) && (A[q] == A[q + 1])) ++q;
nxt[p = 2] = q - 1;
q = std::max(q, 2);
} else {
while ((q <= x) && (q <= y) && (A[q] == B[q])) ++q;
p = 1;
qw(--q, ' ', true);
}
for (int i = pt ? 2 : 3, _ans; i <= x; ++i) {
int a = i - p + 1;
int len = nxt[a];
if ((i + len - 1) >= q) {
_ans = std::max(0, q - i + 1);
while ((q < x) && (_ans < y) && (A[q+1] == B[_ans+1])) {
++_ans; ++q;
}
q = std::max(p = i, q);
} else {
_ans = len;
}
if (pt) {
qw(_ans, ' ', true);
} else {
nxt[i] = _ans;
}
}
}

【字符串】 Z-algorithm的更多相关文章

  1. ExKMP(Z Algorithm) 讲解

    目录 问题引入 CaiOJ 1461 [EXKMP]最长共同前缀长度 算法讲解 匹配过程 next 的求解 复杂度证明 代码解决 一些例题 UOJ #5. [NOI2014]动物园 CF1051E V ...

  2. [leetcode]6. ZigZag Conversion字符串Z形排列

    The string "PAYPALISHIRING" is written in a zigzag pattern on a given number of rows like ...

  3. IE6/7/8中parseInt第一个参数为非法八进制字符串且第二个参数不传时返回值为0

    JavaScript中数字有十进制.八进制.十六进制.以"0"开头的是八进制,"0x"或"0X"开头的是十六进制. parseInt用来把字 ...

  4. Python小代码_7_字符串的字符次数统计

    生成包含 1000 个随机字符的字符串,并统计每个字符出现的次数. import string import random #获取字符 x = string.ascii_letters + strin ...

  5. Python:字符串操作总结

    所有标准的序列操作(索引.分片.乘法.判断成员资格.求长度.取最小值最大值)对字符串同样适用,且字符串是不可变的. 一.字符串格式化 转换说明符 [注]: 这些项的顺序至关重要 (1)%字符:标记转换 ...

  6. Sqlite数据库字符串处理函数replace

    Sqlite 字符串处理函数replace官方说明: replace(X,Y,Z) The replace(X,Y,Z) function returns a string formed by sub ...

  7. Python3简明教程(七)—— 字符串

    字符串是 Python 中最常用的数据类型.本节实验将会学习如何对 Python3 的字符串进行处理操作. 字符串的三种表示 可以通过几种不同的方式表示字符串.如单引号('...')或双引号(&quo ...

  8. java中String字符串

    一.定义String字符串 String字符串和char字符不同,char使用单引号,只能表示一个字符,字符串就是一段文本.String是个类.这个类使用final修饰,所以这个类是不可以继承扩充和修 ...

  9. leetcode 467. 环绕字符串中唯一的子字符串

    题目描述: 把字符串 s 看作是“abcdefghijklmnopqrstuvwxyz”的无限环绕字符串,所以 s 看起来是这样的:"...zabcdefghijklmnopqrstuvwx ...

  10. 后缀自动机----一种将字符串变成DAG的方法

    后缀自动机 (suffix automaton, SAM) 是一个能解决许多字符串相关问题的有力的数据结构.(否则我们也不会用它) 举几个例子,以下的字符串问题都可以在线性时间内通过 SAM 解决 1 ...

随机推荐

  1. Hbase Filter之PrefixFilter

    PrefixFilter PrefixFilter是将rowkey前缀为指定字符串的数据全部过滤出来并返回给用户.例如: Scan scan = new Scan(); scan.setFilter( ...

  2. 《 .NET并发编程实战》实战习题集 - 1 - 隔离副作用

    先发表生成URL以印在书里面.等书籍正式出版销售后会公开内容.

  3. 2019-11-29-VisualStudio-使用三个方法启动最新-C#-功能

    原文:2019-11-29-VisualStudio-使用三个方法启动最新-C#-功能 title author date CreateTime categories VisualStudio 使用三 ...

  4. js 固定div 不随着滚动条滚动

    css .fixed { position: fixed; top:; } javascript function my$(id) { return document.getElementById(i ...

  5. C#使用post方式提交json数据

    尝试了一天,尝试了各种方法,一下方法最直接方便. //地址 string _url = "https://www.dXXXayup.ink/api/User/Login"; //j ...

  6. .net core ajax使用EPPlus上传excle导入总结

    前端 <form class="layui-form" id="div_imp" style="display:none;"> ...

  7. knockout.js绑定(enable,disable,visable)

    <input type="text" data-bind="disable:IsNew" /> enable :是否可用,为true时,可编辑 di ...

  8. java 手写 jvm高性能缓存

    java 手写 jvm高性能缓存,键值对存储,队列存储,存储超时设置 缓存接口 package com.ws.commons.cache; import java.util.function.Func ...

  9. @Async源码探究

    1. @Async源码探究 1.1. 上代码 @SpringBootApplication @EnableAsync public class SpringbootLearnApplication { ...

  10. 如何在相同的类名中单独为选中元素设置JS

    很多时候,我发现对一个类名添加事件,每次都是所有同类名元素一起触发,使用 this可以仅对当前选中的元素应用事件 如 $('.guowai button').click(function() { /* ...