nefu暑假集训5 KMP 个人模板+例题汇总

news2025/1/12 6:46:22

前言:

   KMP算法用于匹配字符串,假设长字符串为s,需要匹配的字符串是p。KMP算法的基础思想是利用一个next[n]数组:next[i]对应的是:以下标i为结尾的连续子串,与以第一个字符开始的子串,相等的最大长度。该数组的作用是,当某一个下标的元素没有匹配成功时,可以通过该长度跳转到与 匹配成功的部分字符串的最后一个,开始下一次匹配。以下是我写的一些题目。

正文:

链接:nefu-KMP - Virtual Judge (vjudge.net)

题目:

A - 剪花布条:

#include<bits/stdc++.h>
using namespace std;
string s,p;
int main(){
    while(cin>>s&&s!="#"){
        cin>>p;
        long long ans=0;
        while(s.find(p)!=-1) s.erase(s.find(p),p.length()),ans++;
        cout<<ans<<endl;
    }
}

这题好像根本用不上KMP,直接用string的find函数就能写出来。

B - Power Strings:

#include<bits/stdc++.h>
using namespace std;
int nxt[1000100];
char s[1000100];
int main(){
    while(~scanf("%s",s+1)&&s[1]!='.') {
        int n=strlen(s+1);
        nxt[1]=0;
        for(int i=2,j=0;i<=n;i++){
            while(j>0&&(j==n||s[i]!=s[j+1])) j=nxt[j];
            if(s[i]==s[j+1]) j++;
            nxt[i]=j;
        }
        if(n%(n-nxt[n])==0&&nxt[n]) printf("%d\n",n/(n-nxt[n]));
        else puts("1");
    }
    return 0;
}

这题实际上就是求字符串最大循环节问题,我们这边直接找到next数组中的next[n],n-next[n]就是他的最小循环子串,但是得判断一下是否可以由该字串直接完美构成(也就是n%(n=next[n])得是0)。

C - KMP:​​​​​​​

 

#include<bits/stdc++.h>
using namespace std;
const int N=1e6+5;
char s1[N],s2[N];
int ne[N];
int main(){
	cin>>s1+1>>s2+1;
	int n=strlen(s1+1),m=strlen(s2+1);
	//cout<<strlen(s1+1)<<" "<<strlen(s2+1)<<endl;
	for(int i=2,j=0;i<=m;i++){
		while(j&&s2[i]!=s2[j+1])j=ne[j];
		if(s2[i]==s2[j+1])j++;
		ne[i]=j;
	}
	for(int i=1,j=0;i<=n;i++){
		while(j&&s1[i]!=s2[j+1])j=ne[j];
		if(s1[i]==s2[j+1])j++;
		if(j==m){
			printf("%d\n",i-m+1);
			j=ne[j];
		}
	}
	for(int i=1;i<=m;i++){
		cout<<ne[i]<<" ";
	}
	return 0;
}

模板题。

D - Radio Transmission:

#include<bits/stdc++.h>
using namespace std;
const int N=1e6+5;
char s[N];
int ne[N];
int main(){
	int l;
	cin>>l>>s+1;
	for(int i=2,j=0;i<=l;i++){
		while(j&&s[i]!=s[j+1])j=ne[j];
		if(s[i]==s[j+1])j++;
		ne[i]=j;
	}
	cout<<l-ne[l]<<endl;
	return 0;
} 

最小循环子串=l-ne[l],证明可以上网搜搜。

E - OKR-Periods of Words:​​​​​​​

这题就是再求最短前缀,我们只需要在求完next数组后直接不断往前循环让next[i]=即可

F - 似乎在梦中见过的样子:​​​​​​​

#include<bits/stdc++.h>
using namespace std;
const int N=150005;
char w[N];
int k,n,ans,ne[N],f[N];
int main(){
	scanf("%s%d",w+1,&k);
	int n=strlen(w+1);
	for(int i=1;i<=n;i++){
		int l=i-1;
		ne[i]=i-1;f[i]=i;
		for(int j=i+1;j<=n;j++){
			while(l!=i-1&&w[l+1]!=w[j])l=ne[l];
			if(w[l+1]==w[j])ne[j]=++l;
			else ne[j]=i-1;
			if(l<i+k-1)f[j]=j;
			else f[j]=f[l];
			ans+=(f[j]<((i+j)>>1));
		}
	}
	cout<<ans<<endl;
	return 0;
}

第一层循环枚举初始位置,第二层循环从每个位置开始做KMP,然后找到满足ABA且A>k的情况。

G - Censoring:​​​​​​​

#include <bits/stdc++.h>
using namespace std;
const int N=1e6+5;
char s1[N],s2[N];
int ne[N],top,pos[N],stk[N];
int main(){
	cin>>s1+1>>s2+1;
	int n=strlen(s1+1),m=strlen(s2+1);
	for(int i=2,j=0;i<=m;i++){
		while(j&&s2[i]!=s2[j+1])j=ne[j];
		if(s2[i]==s2[j+1])j++;
		ne[i]=j;
	}
	for(int i=1,j=0;i<=n;i++){
		stk[++top]=i;
		while(j&&s1[i]!=s2[j+1])j=ne[j];
		if(s1[i]==s2[j+1])j++;
		pos[i]=j;
		if(j==m){
			top-=m;
			j=pos[stk[top]];
		}
	}
	for(int i=1;i<=top;i++){
		cout<<s1[stk[i]];
	}
	return 0;
}

用另外一个数组来模拟删除的过程,其中通过top来模拟数组的最后一位。

H - Compress Words:​​​​​​​

 

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const ll maxn=1000010;
ll n,len,anslen,top,minn,kmp[maxn];
char c[maxn],ans[maxn];
int main(){ 
	cin>>n;
	for(int i=1;i<=n;i++){
		scanf("%s",c+1);
		len=strlen(c+1);
		minn=min(anslen,len);
		top=len;
		c[++top]='#';
		for(int j=1;j<=minn;j++)c[++top]=ans[anslen-(minn-j)];
		for(int j=1;j<top;j++){
			ll k=kmp[j];
			while(k&&c[k+1]!=c[j+1])k=kmp[k];
			if(c[k+1]==c[j+1])k++;
			kmp[j+1]=k;
		}
		for(int j=kmp[top]+1;j<=len;j++)ans[++anslen]=c[j];
	}
	for(int i=1;i<=anslen;i++)cout<<ans[i];
	return 0;
}

I - 动物园:​​​​​​​

 

#include<bits/stdc++.h>
using namespace std;
const int N=1e6+5;
const int mod=1e9+7;
typedef long long ll;
int ne[N],ans[N];
int main(){
	int t;
	cin>>t;
	while(t--){
		ll res=1;
		memset(ne,0,sizeof(ne));
		string s;
		cin>>s;
		int n=s.size();
		ans[0]=0;ans[1]=1;
		for(int i=1,j=0;i<=n;i++){
			while(j&&s[i]!=s[j])j=ne[j];
			if(s[i]==s[j])j++;
			ne[i+1]=j;ans[i+1]=ans[j]+1;	
		}
		for(int i=1,j=0;i<=n;i++){
			while(j&&s[i]!=s[j])j=ne[j];
			if(s[i]==s[j])j++;
			while(2*j>(i+1))j=ne[j];
			res=(res*(ll)(ans[j]+1)%mod);
		}
		cout<<res<<endl;
	}
	return 0;
}

J - Sza-Template:(待补)​​​​​​​

后记:

  KMP还是十分考察思维的算法啊。明天要分方向比赛了,好好写点题吧。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2105218.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

UE 【材质编辑】自定义ShadingMode

【UE 4.27.2】 在UE中提供了多种多样的ShadingMode&#xff0c;相当于一种风格化的处理方案(整体全面的流程调整)&#xff0c;切换ShadingMode可以看到不同的显示效果&#xff1a; 通过简单的拓展&#xff0c;我们可以实现自定义的ShadingMode&#xff0c;使得我们切换到自己的…

JMeter:如何定制Http请求取样器

一般使用JMeter发送HTTP请求时都会用到HTTP Request取样器&#xff0c;这种取样器大多数时候能够满足压力测试的需要。 图1 但也有一些场景&#xff0c;可能需要更加强大的取样器&#xff0c;或者需要定制一些功能&#xff0c;这时就需要自己手动编写取样器。幸好JMeter为我们提…

leveldb源码剖析(二)——LSM Tree

LSM Tree LSM Tree&#xff1a;Log-Structured Merge Tree&#xff0c;日志结构合并树。是一种频繁写性能很高的数据结构。 LSM Tree将写入操作与合并操作分离&#xff0c;数据首先写入磁盘中的日志文件&#xff08;WAL&#xff09;&#xff0c;随后写入内存缓存&#xff0c;…

Android经典实战之Textview文字设置不同颜色、下划线、加粗、超链接等效果

本文首发于公众号“AntDream”&#xff0c;欢迎微信搜索“AntDream”或扫描文章底部二维码关注&#xff0c;和我一起每天进步一点点 SpannableString 在 Android 开发中是一个非常强大的工具&#xff0c;它允许你在单个字符串范围内应用多种样式。使用 SpannableString&#xf…

【 C++ 】 类和对象的学习 (二)

&#x1f618;我的主页&#xff1a;OMGmyhair-CSDN博客 目录 I、类的默认成员函数 一、构造函数 二、析构函数 三、拷贝构造函数 四、 运算符重载 赋值运算符重载 五、取地址重载_普通对象 六、取地址重载_const对象 I、类的默认成员函数 用户没有显示实现&#xff0…

Linux学习笔记5 值得一读,Linux(ubuntu)软件管理,搜索下载安装卸载全部搞定!(上)

本文记录Ubuntu操作系统的软件包管理。 一、背景 整个Linux系统就是大大小小的软件包构成的&#xff0c;在linux系统中&#xff0c;软件的管理非常重要&#xff0c;与其他操作系统不同&#xff0c;linux的软件包管理比较复杂&#xff0c;有时还需要处理软件包之间的冲突。本文…

Python | 泰勒图

写在前面 最近&#xff0c;开始对于CMIP6的一些数据进行评估。Talor图是一个很好展现模式间误差的方式&#xff0c;这里简单记录一下在python中的实现方式。 主要为半图的画法 参考的代码为&#xff1a; https://zenodo.org/records/5548061 效果大致下面这个样子 这边在原…

maven中如何配置多个仓库使其同时生效

场景 有一个项目&#xff0c;我把代码跟本地maven依赖包从同事那里拷贝过来&#xff0c;然后打包却一直打不了&#xff0c;一直报aliyun仓库找不到这个依赖的错误&#xff0c;无论我改成引用本地仓库还是线上aliyun仓库都不行。 依赖 <dependency><groupId>org.spr…

有三层交换机就不用路由器了?真的假的

号主&#xff1a;老杨丨11年资深网络工程师&#xff0c;更多网工提升干货&#xff0c;请关注公众号&#xff1a;网络工程师俱乐部 晚上好&#xff0c;我的网工朋友。 在现代企业网络环境中&#xff0c;三层交换机因其高效的数据包处理能力和较低的延迟而受到广泛欢迎。 然而&…

Python 从入门到实战7(元组)

我们的目标是&#xff1a;通过这一套资料学习下来&#xff0c;通过熟练掌握python基础&#xff0c;然后结合经典实例、实践相结合&#xff0c;使我们完全掌握python&#xff0c;并做到独立完成项目开发的能力。 之前的文章我们通过举例学习了python 中列表的定义及相关操作。今…

Echarts大屏可视化

构建可视化大屏&#xff1a; 构建布局&#xff1a;通过css和html对整个页面进行模块拆分&#xff0c;控制好每一张图的位置和大小&#xff0c;再将echarts实例化对象放到不同的盒子里 效果图&#xff1a; 代码&#xff1a; <!DOCTYPE html> <html lang"en&quo…

斯普林格-《土木工程与结构抗震设计》 Springer-Civil Engineering and Structural Seismic Design

文章目录 一、会议详情二、重要信息三、大会介绍四、出席嘉宾五、征稿主题六、咨询 一、会议详情 二、重要信息 大会官网&#xff1a;https://ais.cn/u/vEbMBz提交检索&#xff1a;EI Compendex、IEEE Xplore、Scopus最终截稿&#xff1a;2024年9月2日23:59 三、大会介绍 四…

关于vue中v-model绑定radio表单元素的说明

在学习中&#xff0c;老师讲在v-model中&#xff0c;绑定的是radio的checked属性&#xff0c;起初看了例子后很不理解&#xff0c;于是开始寻找答案 老师所说的绑定关系 老师给的绑定代码&#xff0c;怎么看来&#xff0c;都不是实例的gender变量绑定radio的checked属性&…

2024 MongoDB 中国用户大会上海站成功举办圆满结束: 技术驱动未来,携手共创辉煌

一年一度 2024 MongoDB 中国用户大会上海站顺利举办&#xff0c;感谢大家的积极参与&#xff01; 在数字化浪潮的背景下&#xff0c;随着人工智能、物联网、5G等前沿技术的快速发展&#xff0c;如何利用这些技术实现业务创新&#xff0c;已成为中国企业在激烈市场竞争中保持领…

Vivado+PetaLinux 系统搭建教程

PetaLinux 是基于 Yocto project DDR SDRAM 双倍数据率同步动态随机存取存储器&#xff08;英语&#xff1a;Double Data Rate Synchronous Dynamic Random Access Memory&#xff0c;简称DDR SDRAM&#xff09;为具有双倍资料传输率的SDRAM&#xff0c;其资料传输速度为系统主…

Matplotlib | 绘制饼图

目录 简介安装 Matplotlib开始绘制简单饼图添加标签添加百分比修改显示方式突出扇形设置标题修改颜色实践&#xff1a;绘制七大洲面积比例图 简介 饼图&#xff08;Pie Chart&#xff09;&#xff0c;用扇形的面积&#xff0c;也就是圆心角的度数来表示数量。 饼图能够十分直…

【Java】ApiPost请求返回 `406` 状态码(jackson)

Java系列文章目录 补充内容 Windows通过SSH连接Linux 第一章 Linux基本命令的学习与Linux历史 文章目录 Java系列文章目录一、前言二、学习内容&#xff1a;三、问题描述3.1 问题截图3.2 错误简介3.2.1 HTTP状态码 406 Not Acceptable3.2.2 序列化和反序列化 3.3 后端问题位置…

yaml文件查看模型的架构

最近在看hrnet模型代码&#xff0c;想查看hrnet的模型架构&#xff0c;输出一下&#xff0c;但是模型参数需要cfg&#xff0c;我就想着怎么把yaml文件导进来然后打印模型呢&#xff0c;直接chat就可以了&#xff0c;下面解释一下每一部分&#xff0c;非常的好理解 yaml文件格式…

传神论文中心|第24期人工智能领域论文推荐

在人工智能领域的快速发展中&#xff0c;我们不断看到令人振奋的技术进步和创新。近期&#xff0c;开放传神&#xff08;OpenCSG&#xff09;传神社区发现了一些值得关注的成就。传神社区本周也为对AI和大模型感兴趣的读者们提供了一些值得一读的研究工作的简要概述以及它们各自…

ChatGPT 3.5/4.0使用手册:解锁人工智能的无限潜能

1. 引言 在人工智能的浪潮中&#xff0c;ChatGPT以其卓越的语言理解和生成能力&#xff0c;成为了一个革命性的工具。它不仅仅是一个聊天机器人&#xff0c;更是一个能够协助我们日常工作、学习和创造的智能伙伴。随着ChatGPT 3.5和4.0版本的推出&#xff0c;其功能和应用范围…