IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> C++算法题 # 25 KMP字符串 -> 正文阅读

[数据结构与算法]C++算法题 # 25 KMP字符串

题目描述
给定一个模式串 S,以及一个模板串 P,所有字符串中只包含大小写英文字母以及阿拉伯数字。

模板串 P 在模式串 S 中多次作为子串出现。

求出模板串 P 在模式串 S 中所有出现的位置的起始下标。

输入格式
第一行输入整数 N,表示字符串 P 的长度。

第二行输入字符串 P。

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

第四行输入字符串 S。

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

数据范围
1≤N≤10^5
1≤M≤10^6
输入样例:

3
aba
5
ababa

输出样例:

0 2

思路
本题的思路先不总结了,然后这是自己参考别人视频理解的,花了太多时间了,以后有时间自己再总结一遍加深印象。这里分享一下我理解KMP时看的一些视频,个人觉得看视频里面的动态变化图更好理解。

KMP算法原理

一、KMP匹配

参考视频:https://www.bilibili.com/video/BV1jb411V78H?spm_id_from=333.1007.top_right_bar_window_history.content.click

KMP 算法永不回退 s数组的指针 i,不走回头路,而是借助 s 数组中储存的信息把 p 移到正确的位置继续匹配,时间复杂度只需 O(N),用空间换时间,所以我认为它是一种动态规划算法。

上面蓝色部分对应s数组,即源串,黄色部分对应p数组,即模式串。

在这里插入图片描述

二、求next数组

参考视频:https://www.bilibili.com/video/BV16X4y137qw?spm_id_from=333.337.search-card.all.click

next数组计算

在这里插入图片描述

next数组的值为模式串当前位置前面子串与相对应主串的最长公共前后缀长度。

代码

在这里插入图片描述

原理解释

在这里插入图片描述

在这里插入图片描述

代码示例
代码参考acwing

#include <iostream>

using namespace std;

const int N = 1e5 + 10;
const int M = 1e6 + 10;

int ne[N];
char p[N], s[M];

int main()
{
	int n, m;
	// p和s下标从1开始
	cin >> n >> p + 1>> m >> s + 1;

	// 求next数组(与kmp类似)
	for(int i = 2, j = 0; i <= n; i++)
	{
		// 这里匹配当前下表i对应j的下一个值是不是相等:如果相等则当前的j加1即可 如果不相等
		// 如果不相等的话,则继续ne[ne[j]]的前面找,知道找到最长前后缀长度;
		while(j && p[i] != p[j+1]) j = ne[j];
		if(p[i] == p[j + 1]) j ++;
		ne[i] = j;

	}
	
	// kmp 匹配过程
	for(int i = 1, j = 0; i <= m; i ++)
	{
		while(j && s[i] != p[j+1]) j = ne[j];
		if(s[i] == p[j + 1]) j ++;
		if(j == n)
		{
			
			cout << i - n  << ' ';
			j = ne[j];
			
		}
	}

	
	return 0;
	
}
  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2022-03-10 22:50:47  更:2022-03-10 22:51:30 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/9 16:54:22-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码