AcWing 831. KMP字符串——算法基础课题解

2024-05-24 01:12

本文主要是介绍AcWing 831. KMP字符串——算法基础课题解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

AcWing 831. KMP 字符串

题目描述

给定一个字符串 𝑆,以及一个模式串 𝑃,所有字符串中只包含大小写英文字母以及阿拉伯数字。

模式串 𝑃 在字符串 𝑆 中多次作为子串出现。

求出模式串 𝑃 在字符串 𝑆 中所有出现的位置的起始下标。

输入格式

第一行输入整数 𝑁,表示字符串 𝑃 的长度。

第二行输入字符串 𝑃。

第三行输入整数 𝑀,表示字符串 𝑆 的长度。

第四行输入字符串 𝑆。

输出格式

共一行,输出所有出现位置的起始下标(下标从 0 开始计数),整数之间用空格隔开。

数据范围

1≤𝑁≤10^5
1≤𝑀≤10^6

输入样例

3
aba
5
ababa

输出样例

0 2

C++

#include <iostream>using namespace std;const int N = 100010, M = 1000010;int n, m;
int nextArr[N];
char text[M], pattern[N];int main() {ios::sync_with_stdio(false);cin.tie(nullptr);cout.tie(nullptr);// 输入模式串长度n和模式串,从索引1开始存储模式串cin >> n >> (pattern + 1) >> m >> (text + 1); // 输入文本串长度m和文本串,从索引1开始存储文本串// 计算 next 数组for (int i = 2, j = 0; i <= n; i++) {// 当字符不匹配时,跳转到前一位置的 next 数组位置while (j && pattern[i] != pattern[j + 1]) j = nextArr[j];// 如果字符匹配,j加1if (pattern[i] == pattern[j + 1]) j++;// 记录 next 数组当前位置的值nextArr[i] = j;}// KMP 匹配过程for (int i = 1, j = 0; i <= m; i++) {// 当字符不匹配时,跳转到前一位置的 next 数组位置while (j && text[i] != pattern[j + 1]) j = nextArr[j];// 如果字符匹配,j加1if (text[i] == pattern[j + 1]) j++;// 如果匹配到整个模式串,输出匹配起始位置if (j == n) {cout << i - n << " ";// 继续查找下一个可能的匹配位置j = nextArr[j];}}return 0;
}

next 数组在 KMP(Knuth-Morris-Pratt)字符串匹配算法中起到关键作用。它的主要目的是在模式匹配过程中,当遇到字符不匹配的情况时,提供一种快速跳转的方法,从而避免重复比较已经匹配过的字符。

具体来说,next 数组的作用有以下几点:

  1. 跳过不必要的比较:当在文本中匹配模式串的过程中遇到字符不匹配时,next 数组可以帮助我们快速确定下一个匹配的起始位置,而不是回退到上一个字符再重新开始匹配。
  2. 提高匹配效率:通过利用next 数组的信息,可以将原本可能需要重复比较的操作优化为线性时间复杂度,从而使得 KMP 算法能够在 O(n + m) 时间内完成匹配任务,其中 n 是文本长度,m 是模式串长度。
  3. 记录部分匹配信息next 数组记录了每个位置之前的最长相同前缀和后缀的长度,这些信息用于在匹配失败时快速确定新的匹配位置。

举例说明next 数组的构建和作用:

假设我们有一个模式串 pattern = “ABABC”。

  • 计算next 数组的步骤如下:
  1. pattern[1] = ‘A’,没有前缀和后缀,所以 next[1] = 0。
  2. pattern[2] = ‘B’,没有前缀和后缀,所以 next[2] = 0。
  3. pattern[3] = ‘A’,前缀"A"和后缀"A"匹配,所以 next[3] = 1。
  4. pattern[4] = ‘B’,前缀"AB"和后缀"AB"匹配,所以 next[4] = 2。
  5. pattern[5] = ‘C’,没有前缀和后缀匹配,所以 next[5] = 0。

构建的 next 数组为 [0, 0, 1, 2, 0]

  • 使用next 数组进行匹配:

当我们在文本中匹配模式串时,如果在某个位置字符不匹配,可以使用next 数组跳转到上一个可能的匹配位置,避免重新比较已经匹配的部分。

Go

package mainimport ("bufio""fmt""os"
)const N = 100010
const M = 1000010var (n, m    intnextArr [N]inttext    [M]bytepattern [N]byte
)func main() {reader := bufio.NewReader(os.Stdin)writer := bufio.NewWriter(os.Stdout)defer writer.Flush()// 读取模式串长度和模式串fmt.Fscanf(reader, "%d\n", &n)patternLine, _ := reader.ReadString('\n')copy(pattern[1:], patternLine)// 读取文本串长度和文本串fmt.Fscanf(reader, "%d\n", &m)textLine, _ := reader.ReadString('\n')copy(text[1:], textLine)// 计算 next 数组for i, j := 2, 0; i <= n; i++ {for j > 0 && pattern[i] != pattern[j+1] {j = nextArr[j]}if pattern[i] == pattern[j+1] {j++}nextArr[i] = j}// KMP 匹配过程for i, j := 1, 0; i <= m; i++ {for j > 0 && text[i] != pattern[j+1] {j = nextArr[j]}if text[i] == pattern[j+1] {j++}if j == n {fmt.Fprintf(writer, "%d ", i-n)j = nextArr[j]}}
}

模板

// s[]是长文本,p[]是模式串,n是s的长度,m是p的长度
求模式串的Next数组:
for (int i = 2, j = 0; i <= m; i ++ )
{while (j && p[i] != p[j + 1]) j = ne[j];if (p[i] == p[j + 1]) j ++ ;ne[i] = j;
}// 匹配
for (int i = 1, j = 0; i <= n; i ++ )
{while (j && s[i] != p[j + 1]) j = ne[j];if (s[i] == p[j + 1]) j ++ ;if (j == m){j = ne[j];// 匹配成功后的逻辑}
}

这篇关于AcWing 831. KMP字符串——算法基础课题解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/996952

相关文章

C#数据结构之字符串(string)详解

《C#数据结构之字符串(string)详解》:本文主要介绍C#数据结构之字符串(string),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录转义字符序列字符串的创建字符串的声明null字符串与空字符串重复单字符字符串的构造字符串的属性和常用方法属性常用方法总结摘

Java实现时间与字符串互相转换详解

《Java实现时间与字符串互相转换详解》这篇文章主要为大家详细介绍了Java中实现时间与字符串互相转换的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、日期格式化为字符串(一)使用预定义格式(二)自定义格式二、字符串解析为日期(一)解析ISO格式字符串(二)解析自定义

SpringBoot实现MD5加盐算法的示例代码

《SpringBoot实现MD5加盐算法的示例代码》加盐算法是一种用于增强密码安全性的技术,本文主要介绍了SpringBoot实现MD5加盐算法的示例代码,文中通过示例代码介绍的非常详细,对大家的学习... 目录一、什么是加盐算法二、如何实现加盐算法2.1 加盐算法代码实现2.2 注册页面中进行密码加盐2.

C#基础之委托详解(Delegate)

《C#基础之委托详解(Delegate)》:本文主要介绍C#基础之委托(Delegate),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1. 委托定义2. 委托实例化3. 多播委托(Multicast Delegates)4. 委托的用途事件处理回调函数LINQ

Java时间轮调度算法的代码实现

《Java时间轮调度算法的代码实现》时间轮是一种高效的定时调度算法,主要用于管理延时任务或周期性任务,它通过一个环形数组(时间轮)和指针来实现,将大量定时任务分摊到固定的时间槽中,极大地降低了时间复杂... 目录1、简述2、时间轮的原理3. 时间轮的实现步骤3.1 定义时间槽3.2 定义时间轮3.3 使用时

python中字符串拼接的几种方法及优缺点对比详解

《python中字符串拼接的几种方法及优缺点对比详解》在Python中,字符串拼接是常见的操作,Python提供了多种方法来拼接字符串,每种方法有其优缺点和适用场景,以下是几种常见的字符串拼接方法,需... 目录1. 使用 + 运算符示例:优缺点:2. 使用&nbsjsp;join() 方法示例:优缺点:3

java字符串数字补齐位数详解

《java字符串数字补齐位数详解》:本文主要介绍java字符串数字补齐位数,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Java字符串数字补齐位数一、使用String.format()方法二、Apache Commons Lang库方法三、Java 11+的St

C++字符串提取和分割的多种方法

《C++字符串提取和分割的多种方法》在C++编程中,字符串处理是一个常见的任务,尤其是在需要从字符串中提取特定数据时,本文将详细探讨如何使用C++标准库中的工具来提取和分割字符串,并分析不同方法的适用... 目录1. 字符串提取的基本方法1.1 使用 std::istringstream 和 >> 操作符示

C语言字符函数和字符串函数示例详解

《C语言字符函数和字符串函数示例详解》本文详细介绍了C语言中字符分类函数、字符转换函数及字符串操作函数的使用方法,并通过示例代码展示了如何实现这些功能,通过这些内容,读者可以深入理解并掌握C语言中的字... 目录一、字符分类函数二、字符转换函数三、strlen的使用和模拟实现3.1strlen函数3.2st

Java反转字符串的五种方法总结

《Java反转字符串的五种方法总结》:本文主要介绍五种在Java中反转字符串的方法,包括使用StringBuilder的reverse()方法、字符数组、自定义StringBuilder方法、直接... 目录前言方法一:使用StringBuilder的reverse()方法方法二:使用字符数组方法三:使用自