kmp——字符串快速匹配

  1. 由来KMP由克努特(Knuth)、莫里斯(Morris)和普拉特(Pratt)共同设计,名字由来取自他们各自名字的首字母。

  1. 改进: 每当一趟匹配过程出现字符不相等时,无需回溯主串的指针,而是利用已经得当的“部分匹配”的结果将模板串向右滑动尽可能远的一段距离,继续进行比较。

  1. 名字前缀:包含首字母,除了最末尾的字符按顺序所构成的所有字符串。如:aabs的前缀有a,aa,aab。

后缀:包含最末尾的字母,除去首字母按顺序所构成的所有字符串。如:aabs的后缀有 s,bs,abs。

最长相等前后缀(自己取的):一般用ne[i]存储,表示有i个字符时的最长相等前后缀。

如a的最长相等前后缀为0,aa为1,aab为0,aaba为1,ababa为3.

  1. 例题

# 【模板】KMP字符串匹配

## 题目描述

给出两个字符串 $s_1$ 和 $s_2$,若 $s_1$ 的区间 $[l, r]$ 子串与 $s_2$ 完全相同,则称 $s_2$ 在 $s_1$ 中出现了,其出现位置为 $l$。

现在请你求出 $s_2$ 在 $s_1$ 中所有出现的位置。

定义一个字符串 $s$ 的 border 为 $s$ 的一个**非 $s$ 本身**的子串 $t$,满足 $t$ 既是 $s$ 的前缀,又是 $s$ 的后缀。

对于 $s_2$,你还需要求出对于其每个前缀 $s'$ 的最长 border $t'$ 的长度。

## 输入格式

第一行为一个字符串,即为 $s_1$。

第二行为一个字符串,即为 $s_2$。

## 输出格式

首先输出若干行,每行一个整数,**按从小到大的顺序**输出 $s_2$ 在 $s_1$ 中出现的位置。

最后一行输出 $|s_2|$ 个整数,第 $i$ 个整数表示 $s_2$ 的长度为 $i$ 的前缀的最长 border 长度。

## 样例 #1

### 样例输入 #1

```

ABABABC

ABA

```

### 样例输出 #1

```

1

3

0 0 1

```

## 提示

### 样例 1 解释

![](https://cdn.luogu.com.cn/upload/pic/2257.png)

对于 $s_2$ 长度为 $3$ 的前缀 `ABA`,字符串 `A` 既是其后缀也是其前缀,且是最长的,因此最长 border 长度为 $1$。

### 数据规模与约定

**本题采用多测试点捆绑测试,共有 3 个子任务**。

- Subtask 1(30 points):$|s_1| \leq 15$,$|s_2| \leq 5$。

- Subtask 2(40 points):$|s_1| \leq 10^4$,$|s_2| \leq 10^2$。

- Subtask 3(30 points):无特殊约定。

对于全部的测试点,保证 $1 \leq |s_1|,|s_2| \leq 10^6$,$s_1, s_2$ 中均只含大写英文字母。

哎呦,第一次用c++!!!

有来有往,才能长久以往,所以点个赞吧。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值