KMP整理+个人推导+快速上手理解

整理了一下KMP的写法:

这个是我自己写的(个人推导,可能在时间复杂度上表现较弱,但是非常帮助初学者进行理解!)

下面是代码, ne 是next数组。我这个next数组表示的是:

ne[i] : 当s[i]和target不匹配的时候,可以向前移动几格

例如

s : abc abc x

target : abc abe abc abc x

当s去匹配target的时候,匹配导了target[6] : e的时候,匹配失败了,那我们不需要仅仅向前移动一格,可以直接移到target[4] 处进行再匹配。

abc abe abc abc x ---> abc abe abc abc x

↑ ↑

abc abc x abc abc x

那么ne[6] = 3表示前进3.

下面先有暴力做法,再给出相对正确做法。

python 复制代码
    ne[1] = 1;
    // for (int i = 1; i < n; i ++ ){
    //     //当i位置不匹配的时候,考虑s.substr(0,i);
    //     //即求出最长前缀和最长后缀的相同情况下的长度
    //     int cur = i;
    //     for (int j = 1; j <= i; j ++ ){
    //         bool f = false;
    //         for (int k = j; k <= i; k ++ ){
    //             if(s[k] != s[k - j]){f = true ; break;}
    //         }
    //         if(!f){cur = j ; break;}
    //     }
    //     ne[i] = cur;
    //     printf("%d\n" , cur);
    // }
    for (int i = 2; i <= n; i ++ ){
        if(s[i] == s[1]){
            int  p = 1;
            ne[i] = ne[i-1];
            while(s[i + p] == s[1 + p]){
                ne[i + p] = ne[i + p - 1];
                p++;
            }
            i += (p - 1);
        }else{
            ne[i] = i;
        }
    }

最优代码:

cpp 复制代码
#include <iostream>

using namespace std;

const int N = 100010, M = 1000010;

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

int main()
{
    cin >> n >> p + 1 >> m >> s + 1;

    for (int i = 2, j = 0; i <= n; 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 <= m; i ++ )
    {
        while (j && s[i] != p[j + 1]) j = ne[j];
        if (s[i] == p[j + 1]) j ++ ;
        if (j == n)
        {
            printf("%d ", i - n);
            j = ne[j];
        }
    }

    return 0;
}
相关推荐
数智工坊14 小时前
【UniT论文阅读】:用统一物理语言打通人类与人形机器人的知识壁垒
论文阅读·人工智能·深度学习·算法·机器人
ABAP 成14 小时前
删除+新增原始BOM工序+订单BOM+工序笔记
笔记
梓䈑14 小时前
【算法题攻略】模拟
c++·算法
Evand J14 小时前
【课题推荐与代码介绍】卡尔曼滤波器正反向估计算法原理与MATLAB实现
开发语言·算法·matlab
XS03010614 小时前
MyBatis基础实战笔记一
笔记·mybatis
DFT计算杂谈14 小时前
VASP新手入门: IVDW 色散修正参数
linux·运维·服务器·python·算法
Oll Correct14 小时前
实验二十五:从IPv4向IPv6过渡所使用的隧道技术
网络·笔记
吃着火锅x唱着歌14 小时前
LeetCode 962.最大宽度坡
算法·leetcode·职场和发展
无限进步_14 小时前
【C++】C++11的类功能增强与STL变化
java·前端·数据结构·c++·后端·算法
WL_Aurora14 小时前
Python 算法基础篇之排序算法(一):冒泡、选择、插入
python·算法·排序算法