【字符串】 Z-algorithm
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\) 的对应匹配位置。
由于所求是公共前缀字串,因此有
\]
引入一组辅助变量,设 \(next_j\) 为 \(T_{j \sim m}\) 与 \(T\) 的最长公共前缀子串长度。
根据定义,有
\]
分两种情况讨论。
第一种情况,\(j + next_j < q\),即 \(T_{j \sim j + next_j}\) 是 \(S_{p \sim q}\) 的字串,因此有
\]
又因为 \(T_{j \sim j + 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的更多相关文章
- ExKMP(Z Algorithm) 讲解
目录 问题引入 CaiOJ 1461 [EXKMP]最长共同前缀长度 算法讲解 匹配过程 next 的求解 复杂度证明 代码解决 一些例题 UOJ #5. [NOI2014]动物园 CF1051E V ...
- [leetcode]6. ZigZag Conversion字符串Z形排列
The string "PAYPALISHIRING" is written in a zigzag pattern on a given number of rows like ...
- IE6/7/8中parseInt第一个参数为非法八进制字符串且第二个参数不传时返回值为0
JavaScript中数字有十进制.八进制.十六进制.以"0"开头的是八进制,"0x"或"0X"开头的是十六进制. parseInt用来把字 ...
- Python小代码_7_字符串的字符次数统计
生成包含 1000 个随机字符的字符串,并统计每个字符出现的次数. import string import random #获取字符 x = string.ascii_letters + strin ...
- Python:字符串操作总结
所有标准的序列操作(索引.分片.乘法.判断成员资格.求长度.取最小值最大值)对字符串同样适用,且字符串是不可变的. 一.字符串格式化 转换说明符 [注]: 这些项的顺序至关重要 (1)%字符:标记转换 ...
- Sqlite数据库字符串处理函数replace
Sqlite 字符串处理函数replace官方说明: replace(X,Y,Z) The replace(X,Y,Z) function returns a string formed by sub ...
- Python3简明教程(七)—— 字符串
字符串是 Python 中最常用的数据类型.本节实验将会学习如何对 Python3 的字符串进行处理操作. 字符串的三种表示 可以通过几种不同的方式表示字符串.如单引号('...')或双引号(&quo ...
- java中String字符串
一.定义String字符串 String字符串和char字符不同,char使用单引号,只能表示一个字符,字符串就是一段文本.String是个类.这个类使用final修饰,所以这个类是不可以继承扩充和修 ...
- leetcode 467. 环绕字符串中唯一的子字符串
题目描述: 把字符串 s 看作是“abcdefghijklmnopqrstuvwxyz”的无限环绕字符串,所以 s 看起来是这样的:"...zabcdefghijklmnopqrstuvwx ...
- 后缀自动机----一种将字符串变成DAG的方法
后缀自动机 (suffix automaton, SAM) 是一个能解决许多字符串相关问题的有力的数据结构.(否则我们也不会用它) 举几个例子,以下的字符串问题都可以在线性时间内通过 SAM 解决 1 ...
随机推荐
- MAST 397B: Introduction to Statistical Computing
MAST 397B: Introduction to Statistical ComputingABSTRACTNotes: (i) This project can be done in group ...
- Scala中sortBy和Spark中sortBy区别
Scala中sortBy是以方法的形式存在的,并且是作用在Array或List集合排序上,并且这个sortBy默认只能升序,除非实现隐式转换或调用reverse方法才能实现降序,Spark中sortB ...
- 浅析libuv源码-node事件轮询解析(1)
好久没写东西了,过了一段咸鱼生活,无意中想起了脉脉上面一句话: 始终保持自己的竞争力.所以,继续开写! 一般的JavaScript源码看的已经没啥意思了,我也不会写什么xx入门新手教程,最终决定还是啃 ...
- 关于wordpress4.8中的Twenty Seventeen主题的主题选项增加章节的实现
我这里的wordpress版本是4.8 默认的主题是 Twenty Seventeen 我想实现的事 主题选项的首页 多增加2个章节 默认是只有4个章节 我想在增加2个 到6个 看下实现后的效果 ...
- 关于thinkphp3.2.3集成phpmailer
关于thinkphp3.2.3集成phpmailer 1 我用的是phpmailer5.1的版本 先把文件解压缩放到这个位置 2 封装到函数里面 function email($email,$tit ...
- 破解压缩包的几种方式(zip伪加密 爆破 CRC32碰撞 已知明文攻击)
zip伪加密 zip文件是由3部分组成,详见文末 压缩源文件数据区+压缩源文件目录区+压缩源文件目录结束标志 在压缩源文件数据区有个2字节的 全局方式位标记 ,在压缩源文件目录区也有个2字节的 全局方 ...
- flink Iterate迭代基本概念
基本概念:在流中创建“反馈(feedback)”循环,通过将一个算子的输出重定向到某个先前的算子.这对于定义不断更新模型的算法特别有用. 迭代的数据流向:DataStream → IterativeS ...
- 跟着9张思维导图学习Javascript js 关键字和保留字 css3中的BFC,IFC,GFC和FFC
跟着9张思维导图学习Javascript 学习的道路就是要不断的总结归纳,好记性不如烂笔头,so,下面将 po 出我收集的 9 张 javascript 相关的思维导图(非原创). 思维导图小ti ...
- Kubernetes基础服务架构图
最近看了一些kubernetes的相关资料, 简单的画了一个原理图 欢迎大家批阅
- php获取ssl验证的https页面的源码
$response = "https://faculty.xidian.edu.cn/system/resource/tsites/tsitesencrypt.jsp?id=_tsites_ ...