学习总结11

KMP算法

全称Knuth-Morris-Pratt算法,是一种字符串匹配算法。该算法的目的是在一个文本串S内查找一个模式串P的出现位置。

KMP算法的核心思想是利用模式串自身的特性来避免不必要的字符比较。算法通过构建一个部分匹配表(也称为next数组),来记录模式串中每个字符之前的前缀子串和后缀子串的最长公共长度。根据这个表,算法可以通过调整模式串的起始位置来跳过不需要比较的字符,从而提高匹配的效率。

KMP算法的具体步骤如下:

  1. 预处理模式串P,构建部分匹配表next数组;

  2. 设置两个指针i和j,分别指向文本串S和模式串P的起始位置;

  3. 逐个比较S[i]和P[j],如果相等,则i和j同时后移;

  4. 如果不相等,根据next数组跳过一部分字符,将j更新为next[j],同时i保持不变;

  5. 重复步骤3和步骤4,直到找到一个匹配或者S已经遍历完。

KMP算法的时间复杂度为O(m+n),其中m和n分别是文本串和模式串的长度。相比于暴力匹配算法的时间复杂度O(m*n),KMP算法能够在较短的时间内找到匹配位置。

题目描述

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

现在请你求出 s2​ 在 s1​ 中所有出现的位置。

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

对于 s2​,你还需要求出对于其每个前缀 s′ 的最长 border t′ 的长度。

输入格式

第一行为一个字符串,即为 s1​。

第二行为一个字符串,即为 s2​。

输出格式

首先输出若干行,每行一个整数,按从小到大的顺序输出 s2​ 在 s1​ 中出现的位置。

最后一行输出 ∣s2​∣ 个整数,第 i 个整数表示 s2​ 的长度为 i 的前缀的最长 border 长度。

输入输出样例

输入 #1复制

ABABABC

ABA

输出 #1复制

1

3

0 0 1

说明/提示

样例 1 解释

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

数据规模与约定

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

Subtask 1(30 points):∣s1​∣≤15,∣s2​∣≤5。

Subtask 2(40 points):∣s1​∣≤10^4,∣s2​∣≤10^2。

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

对于全部的测试点,保证 1≤∣s1​∣,∣s2​∣≤10^6,s1​,s2​ 中均只含大写英文字母。

cs 复制代码
#include<stdio.h> 
#include<string.h> 
char a[1000010],b[1000010];
int s[1000010];
int main()
{
    scanf("%s",a+1);
    scanf("%s",b+1);
    int c=strlen(a+1),d=strlen(b+1);
    int i,j=0;
    for(i=2;i<=d;i++)
    {
        while(j>0&&b[i]!=b[j+1]) 
            j=s[j];
        if(b[i]==b[j+1]) 
            j++;
        s[i]=j;
    }
    j=0;
    for(i=1;i<=c;i++)
    {
        while(j>0&&a[i]!=b[j+1]) 
            j=s[j];
        if(a[i]==b[j+1]) 
            j++;
        if(j==d) 
            printf("%d\n",i-d+1),j=s[j];
    }
    for(int i=1;i<d;i++)
        printf("%d ",s[i]);
    printf("%d",s[d]);
}
相关推荐
长安er6 小时前
LeetCode215/347/295 堆相关理论与题目
java·数据结构·算法·leetcode·
元亓亓亓6 小时前
LeetCode热题100--62. 不同路径--中等
算法·leetcode·职场和发展
小白菜又菜7 小时前
Leetcode 1925. Count Square Sum Triples
算法·leetcode
粉红色回忆7 小时前
用链表实现了简单版本的malloc/free函数
数据结构·c++
登山人在路上8 小时前
Nginx三种会话保持算法对比
算法·哈希算法·散列表
写代码的小球8 小时前
C++计算器(学生版)
c++·算法
AI科技星8 小时前
张祥前统一场论宇宙大统一方程的求导验证
服务器·人工智能·科技·线性代数·算法·生活
予枫的编程笔记8 小时前
Redis 核心数据结构深度解密:从基础命令到源码架构
java·数据结构·数据库·redis·缓存·架构
wadesir9 小时前
掌握Rust并发数据结构(从零开始构建线程安全的多线程应用)
数据结构·安全·rust
Fuly10249 小时前
大模型剪枝(Pruning)技术简介
算法·机器学习·剪枝