关于KMP
KMP其实是三个人名字的缩写,因为是他们同时发现的(大佬惹不起);
KMP作为CSP考点,主要亮点是其优秀的匹配复杂度,而且消耗空间小,比起hash虽然有些局限性,但是因为其正确率高,所以经常被人使用.
前置知识
关于字符串的读取,以及字符串相关操作的基础了解,这里涉及字符串匹配以及子串;
入坑
其实KMP并不困难,只是让人难受的是它比较抽象的数组跳跃,我想这个并不需要过多解释;
思想
KMP常用于一个字符串是否出现在另一个字符串中.我们知道,如果暴力匹配了话,每次失配时就必须重新开始(不能贪心地从失配位置匹配),这样造成很大的浪费,那么我们想从已经匹配过的字符串中提取一些信息,以至于让我们不跳那么远,那这怎么办?
KMP算法就由此诞生了,它通过记录模式串的内部信息,为匹配时提供信息,可以节省大量时间.
模版
1 |
|
关于next数组的几个性质
因为next与stl冲突所以命名为nxt数组;
next数组有一些性质:
$next[l]$ 到 $l$ 为模式串的最小循环节,当然必须满足一个条件,即最小循环节长度是整个串长度的因数,如果不是了话,那么一定是开头的字符串有残余,而残余字符串为循环节的后缀;
那么考虑一下,如果我们想要找最小循环节,直接初始化后,找 $next[l]$ 即可,当然还要判断一下;
想象一下 $next$ 数组的跳跃,我们能找到什么?即从 $1$ ~ $next [ l ] $ 既是前缀又是后缀,那么我们可以找到子串中的最大前缀和后缀相同的;
匹配时需要注意的细节
我们常常会遇到让我们求出循环次数,以及不重叠循环次数,其区别只是判断 $j==l2$ 时 $j$ 是否要跳回 $next[j]$ ;
或者是直接判断是否有这个模式串,直接 $return$ 即可;
关于题目变形
主要是应该看出匹配方式,以及字符串的重构问题;