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. ng 引入query

    ng 引入jquery 1.在项目中 npm install --save jquery 在对应组件中加入 import * as $ from "jquery";   在angu ...

  2. Linux 安装Mysql1.8【yum安装】

    .下载mysql的yum源 wget http://dev.mysql.com/get/mysql57-community-release-el7-8.noarch.rpm2.安装yum源yum lo ...

  3. 解决mac/win双系统,mac原生读写NTFS分区重启后失效的问题

    安装mac/win双系统,然后在mac下启用原生的NTFS分区读写功能,并将分区创建桌面快捷方式后,会发现有时候进入win后再进mac,原来创建的分区桌面快捷方式是白色的图标,并且分区也无法打开,这个 ...

  4. SSM基本配置详解

    需要查看SSM基本依赖和完整配置文件的到:SSM基本配置及依赖 示例项目:SSMDemo 1 Spring IOC容器配置 1.1 applicationContext.xml 1.1.1 配置数据源 ...

  5. Java学习:Stream流式思想

    Stream流 Java 8 API添加了一种新的机制——Stream(流).Stream和IO流不是一回事. 流式思想:像生产流水线一样,一个操作接一个操作. 使用Stream流的步骤:数据源→转换 ...

  6. Dozer JAVA的POJO 映射工具

    Dozerhttp://www.manongjc.com/article/50949.html JAVA的映射工具 BeanUtils dozer的使用方法https://blog.csdn.net/ ...

  7. C# 中 ==和equals的区别

    不想说太多,直接上代码,这两个就没什么联系,有自己独立的规则.联系在一起其实不利于记忆. 下面是测试代码 Console.WriteLine("--equals和==的区别--") ...

  8. python高级编程——锁

    锁           在使用用的过程中需要导入threading模块的Lock类 使用锁: 当多个线程几乎同时修改某一个共享数据的时候,需要进行同步控制 线程同步能够保证多个线程安全访问竞争资源,最 ...

  9. 【夯实基础】-浅谈"单点登录"的几种实现方式

    单点登录 一.Session跨域 所谓Session跨域就是摒弃了系统提供的Session,而使用自定义的类似Session的机制来保存客户端数据的一种解决方案. 如:通过设置cookie的domai ...

  10. vue 利用路由跨页传参

    第一页,点击进入第二页进行传值: <template> <div id="app"> <div><router-link to=" ...