【字符串】KMP字符串匹配
Definition
算法是一个字符串匹配算法。他接收两个字符串,返回在中出现的所有位置。
以下称需要被匹配的串为主串,可能在主串中多次出现的串为模式串。约定主串的长度为,模式串的长度为。
朴素的想法显然是对使用两个指针分别指向。枚举的位置,暴力右移观察是否匹配。复杂度。
考虑当一个位置失配的时候,如果主串的当前位置与模式串前面的某些位置相同,则可以直接跳到前面的位置,而不用从头开始。
举个例子:
主串:aaabbbc
模式串:aabbbc。
从主串第一个位置开始匹配,当时,发现与失配,但是考虑完全不需要从第二个位置令重新匹配,而是直接令等于可以与匹配的上一个前缀,即可重新匹配。所谓上一个前缀,设被赋值为,则满足主串[i-ki]==模式串[1k]。我们可以使用一个数组直接预处理这个前缀,然后直接在失配时令模式串指针向前按照数组向前跳即可。
考虑数组的求法:把模式串错一位与自身匹配。j跳到的位置显然就是当前位置的。
Example
Description
如题,给出两个字符串s1和s2,其中s2为s1的子串,求出s2在s1中所有出现的位置。
为了减少骗分的情况,接下来还要输出子串的前缀数组next。
Input
两行分别是和
Output
若干行,每行包含一个整数,表示s2在s1中出现的位置
接下来1行,包括length(s2)个整数,表示前缀数组next[i]的值。
Hint
字符串长度不超过
Code
#include<cstdio>
#include<cstring>
#define rg register
#define ci const int
#define cl const long long
typedef long long int ll;
template <typename T>
inline void qr(T &x) {
rg char ch=getchar(),lst=' ';
while((ch > '9') || (ch < '0')) lst=ch,ch=getchar();
while((ch >= '0') && (ch <= '9')) x=(x<<1)+(x<<3)+(ch^48),ch=getchar();
if(lst == '-') x=-x;
}
namespace IO {
char buf[120];
}
template <typename T>
inline void qw(T x,const char aft,const bool pt) {
if(x < 0) {x=-x,putchar('-');}
rg int top=0;
do {IO::buf[++top]=x%10+'0';} while(x/=10);
while(top) putchar(IO::buf[top--]);
if(pt) putchar(aft);
}
template <typename T>
inline T mmax(const T a,const T b) {return a > b ? a : b;}
template <typename T>
inline T mmin(const T a,const T b) {return a < b ? a : b;}
template <typename T>
inline T mabs(const T a) {return a < 0 ? -a : a;}
template <typename T>
inline void mswap(T &_a,T &_b) {
T _temp=_a;_a=_b;_b=_temp;
}
const int maxn = 1000010;
char s1[maxn],s2[maxn];
int nxt[maxn];
void KMP(char *a,char *b,int l1,int l2,bool pt) {
rg int i,j=0;
for(i=pt?1:2;i<=l1;++i) {
while(j&&(b[j+1] != a[i])) j=nxt[j];
if(b[j+1] == a[i]) ++j;
if(!pt) nxt[i]=j;
if(j == l2) {
qw(i-l2+1,'\n',true);
}
}
}
int main() {
scanf("%s\n%s",s1+1,s2+1);
int l1=strlen(s1+1),l2=strlen(s2+1);
KMP(s2,s2,l2,l2,false);
KMP(s1,s2,l1,l2,true);
for(rg int i=1;i<l2;++i) qw(nxt[i],' ',true);
qw(nxt[l2],'\n',true);
return 0;
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· 开发者必知的日志记录最佳实践
· winform 绘制太阳,地球,月球 运作规律
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· AI 智能体引爆开源社区「GitHub 热点速览」
· Manus的开源复刻OpenManus初探
· 写一个简单的SQL生成工具