KMP 模式串匹配 失去匹配的瞬间你还有什么
KMP:
KMP算法是一种改进的字符串匹配算法,由D.E.Knuth,J.H.Morris和V.R.Pratt同时发现,因此人们称它为克努特——莫里斯——普拉特操作(简称KMP算法)。KMP算法的关键是利用匹配失败后的信息,尽量减少模式串与主串的匹配次数以达到快速匹配的目的。具体实现就是实现一个next()函数,函数本身包含了模式串的局部匹配信息。时间复杂度O(m+n)。
——百度百科。
自我理解:
kmp算法最最最最核心的思想,就是在每一次失去匹配的时候,找到最大的可能能够匹配的子段进行匹配。
也就是著名的nxt数组。
算法流程:
数组只介绍一个nxt数组。nxt[i]表示,从1~i前缀S中,最长的前缀等于后缀的长度(不能是S)。
在每次失去匹配的时候,因为上次已经能够匹配到j了,所以1~j和 i-j~i-1 是相等的。
所以我们让j=nxt[j]的时候,根据定义,1~nxt[j] = j-nxt[j]+1~j , 又因为:1~j = i-j~i-1 所以 j-nxt[j]+1 ~ j = i-nxt[j] ~ i-1
这样,我们退一步到nxt,可以找到失去匹配后,最大的可能再次匹配上的字段长度nxt[j]
代码实现:
void kmp(){
nxt[]=;
for(int i=,j=;i<=l2;i++){
while(j>&&b[i]!=b[j+]) j=nxt[j];
if(b[i]==b[j+]) j++;
nxt[i]=j;
}
}
下面这里还加上了f数组,f[i]表示,在a串中,以第i位结尾的所有子串,和b串的前缀最大匹配的长度。
转移是类似的。
void fin(){
for(int i=,j=;i<=l1;i++){
while(j>&&(j==l2||a[i]!=b[j+])) j=nxt[j];
if(a[i]==b[j+]) j++;
f[i]=j;
if(f[i]==l2){
printf("%d\n",i-l2+);
}
}
}
这样就可以求出模式串在原始串中出现的位置和个数了。
应用:
1.模式串在主串中出现的次数。见上述代码。
2.求一个串的循环节:
长度为n的字符串的最短循环节是:n-nxt[n],(可以证明:1.可以循环 2.是最短的)
当n%(n-nxt[n])等于0的时候,字符串是一个循环字符串。最长循环次数为:n/(n-nxt[n])
应用例题:
T1:NOI2014 动物园
这个题考察S的前缀中,处理不重叠前缀等于后缀的数量。
可以先求出重叠的nxt[i],num1[]
在处理不重叠的num2[]的时候,跳nxt[j]的时候,跳跃不停止的条件加上一个2*j>i还要跳,因为不能重叠。
跳完后,判断能否加1,如果2*j>i再跳一次。(WA了。。。)
最后,num2[i]=num1[j]+1 注意,这里一定是num1,也就是可重叠的。因为2*j<=i,所以,j中重叠是没有关系的。(WA了。。。)
加一表示1~j和i-j+1~i也是一个。
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N=+;
const int mod=1e9+;
int nxt1[N],nxt2[N],num1[N],num2[N];//1: can 2: can't folded
char a[N];
int l;
ll ans;
void kmp1(){
nxt1[]=,num1[]=;
for(int i=,j=;i<=l;i++){
while(j&&a[i]!=a[j+]) j=nxt1[j];
if(a[i]==a[j+]) j++;
nxt1[i]=j;
if(nxt1[i]) num1[i]=num1[j]+;
}
}
void kmp2(){
nxt2[]=;num2[]=;
for(int i=,j=;i<=l;i++){
while(j&&(*j>i||a[i]!=a[j+])) j=nxt1[j];//warning!!
if(a[i]==a[j+]) j++;
if(*j>i) j=nxt1[j];//warning!!
nxt2[i]=j;
if(nxt2[i]) num2[i]=num1[j]+;
}
}
int main()
{
int n;
scanf("%d",&n);
while(n--){
scanf("%s",a+);
ans=;
l=strlen(a+);
kmp1();
kmp2();
for(int i=;i<=l;i++){
ans=(ans*((ll)+num2[i]))%mod;
}
printf("%lld\n",ans);
memset(num1,,sizeof num1);
memset(num2,,sizeof num2);
}
return ;
}
动物园
T2:bzoj4641 基因改造
KMP 模式串匹配 失去匹配的瞬间你还有什么的更多相关文章
- kmp模式串匹配
大二的时候百度看不懂,现在大三下学期了百度也看不懂.(实在是不能理解这个next数组究竟是怎么样工作的,为什么会得出那样的结果.)如果有好心人知道的话希望可以联系我告诉我(邮箱:4609019410@ ...
- 模式串匹配KMP详解
关于KMP模式串匹配网上蛮多的. 对于KMP有自己理解所以写下来希望能够对你们的学习有帮助. 之前暑假的时候学过,然后好长时间没用发现又忘了,现在再看看发现有了新的理解. ============== ...
- hdu 1686 Oulipo 【KMP】(计算模式串匹配的次数——与已匹配的字串可以有交集)
题目链接:https://vjudge.net/contest/220679#problem/B 题目大意: 输入一个T,表示有T组测试数据: 每组测试数据包括一个字符串W,T,T长度大于W小于100 ...
- 单模式串匹配----浅谈kmp算法
模式串匹配,顾名思义,就是看一个串是否在另一个串中出现,出现了几次,在哪个位置出现: p.s. 模式串是前者,并且,我们称后一个 (也就是被匹配的串)为文本串: 在这篇博客的代码里,s1均为文本串, ...
- 模式串匹配之KMP算法
模式串匹配之KMP算法 KMP算法 模式值计算(next[j]) (1) next[0]=-1, 第一个字符模式值为-1 (2) next[j]=-1, T中下标为j的字符与首字符相同,且j前面的1 ...
- C++编程练习(7)----“KMP模式匹配算法“字符串匹配
子串在主串中的定位操作通常称做串的模式匹配. KMP模式匹配算法实现: /* Index_KMP.h头文件 */ #include<string> #include<sstream& ...
- HDU-2087-剪花布条 【KMP】(求模式串的匹配个数——与已匹配的字串不交)
题目链接:https://vjudge.net/contest/220679#problem/C 剪花布条 ...
- AC自动机——多模式串匹配的算法思想
标准KMP算法用于单一模式串的匹配,即在母串中寻求一个模式串的匹配,但是现在又存在这样的一个问题,如果同时给出多个模式串,要求找到这一系列模式串在母串存在的匹配个数,我们应该如何处理呢? 基于KMP算 ...
- POJ 3167 Cow Patterns(模式串浮动匹配)
题目链接:http://poj.org/problem?id=3167 题意:模式串可以浮动的模式匹配问题给出模式串的相对大小,需要找出模式串匹配次数和位置. 思路:统计比当前数小,和于当前数相等的, ...
随机推荐
- 将 C# 枚举序列化为 JSON 字符串 实践
一.定义枚举 public enum SiteTypeEnum { 中转部 = 1, 网点 = 2 } 还有 BooleanEnum 和 OptTypeEnum 这两个枚举,这里暂且省略了它们的定义. ...
- LVS负载均衡下session共享的实现方式-持久化连接
之前简单介绍LVS负载均衡的高可用方案实施,下面详细说明LVS的session解决方案: LVS算法中,SH算法可以实现将同一客户端的请求总是发送给第一次指定的RS,除非该RS出现故障不能再提供服务. ...
- Python初始编码-3
01010100 新11010000 开11010100 一01100000 家11000000 看11000000 看 01010100011101110101011110110A B C01000 ...
- maven依赖的描述
maven的坐标和依赖 坐标和依赖,主要涉及的就是pom文件的头部和<dependencies>标签部分(1)pom文件的头部 这里头部不是指pom文件的开头<project> ...
- shell脚本--分支、条件判断
在看选择判断结构之前,请务必先看一下数值比较与文件测试 if....else... #!/bin/bash #文件名:test.sh score=66 # //格式一 if [ $score -lt ...
- Vue.directive注册指令
指令定义函数提供了几个钩子函数(可选): vue指令的生命周期 bind: 只调用一次,指令第一次绑定到元素时调用,用这个钩子函数可以定义一个在绑定时执行一次的初始化动作. inserted: 被绑定 ...
- [转帖]从1G到5G
一文看懂无线通信:从1G到5G 投递人 itwriter 发布于 2019-02-03 15:25 评论(2) 有280人阅读 [收藏] « » 文/老和山下的小学僧 最近太邪乎,先引用一个马克思政治 ...
- Mordern Effective C++ --auto
5. 优先使用auto而非显示类型声明 在C++之中,使用auto关键字声明类型可以将程序员从输入繁琐的类型中解放出来,编译器会自动推导出变量的实际类型. template<typename I ...
- OneZero第七周第一次站立会议(2016.5.9)
1. 时间: 12:15--12:25 共计10分钟. 2. 成员: X 夏一鸣 * 组长 (博客:http://www.cnblogs.com/xiaym896/), G 郭又铭 (博客:http ...
- hive-site.xml
https://cwiki.apache.org/confluence/display/Hive/AdminManual+MetastoreAdmin#AdminManualMetastoreAdmi ...