纯小白蓝桥杯备赛笔记--DAY10(字符串)

news2025/1/16 6:55:28

文章目录

    • KMP字符串哈希
      • 算法简介:
      • 斤斤计较的小z--2047
      • 字符串hash
    • Manacher
      • 回文串的性质
      • 算法简介
      • 最长回文子串
    • 字典树基础
      • 朴素字符串查找
      • 步骤
      • 前缀判定--1204
    • 01tire
      • 算法简介:
      • 例题1:
      • 例题2:

KMP字符串哈希

算法简介:

在这里插入图片描述

  • 真前后缀的意义:前后缀不相等。
  • 注意方向都是正向的,而不是回文的字符串。
  • 在这里插入图片描述
  • 模版:
    在这里插入图片描述
char s[N],p[M];
int nex[M];
int n=strlen(s+1),m=strlen(p+1);//字符串的下标从1开始
nex[0]=nex[1]=0;
for(int i=2,j=0;i<m;i++)
{
	//不断匹配p[i]和p[j+1]
	while(j&&p[i]!=p[j+1])j=nex[j];
	if(p[i]==p[j+1])j++;//从while出来后要么j=0,要么匹配成功
	nex[i]=j; 
 } 
  • 用nex数组去匹配s
for(int i=1,j=0;i<=n;i++)
{
while(j&&s[i]!=p[j+1])j=nex[j];//失配时移动
	if(s[i]==p[j+1])j++;//成功匹配一个字符
	if(j==m)//成功匹配一次 
}

斤斤计较的小z–2047

在这里插入图片描述

#include<bits/stdc++.h>
using namespace std;

const int N=1e6+9;
char s[N],p[N];
int nex[N];
int main()
{
	ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
	cin>>p+1;int m=strlen(p+1);//模式串 
	cin>>s+1;int n=strlen(s+1);//文本串
	
	//get next
	nex[0]=nex[1]=0;
	for(int i=2,j=0;i<=m;i++)
	{
		while(j&&p[i]!=p[j+1])j=nex[j];
		if(p[i]==p[j+1])j++;
		nex[i]=j;
	 } 
	 
	 //对s串进行匹配
	 int ans=0;
	  for(int i=1,j=0;i<=n;i++)
	{
		while(j&&s[i]!=p[j+1])j=nex[j];
		if(s[i]==p[j+1])j++;
		if(j==m)ans++;
	 } 
	 cout<<ans<<endl;
	 return 0;
	 
 } 

定义了一个字符数组s和p,分别用于存储文本串和模式串。同时定义了一个整数数组nex,用于存储模式串的next数组。
通过cin读取输入的模式串和文本串,并计算它们的长度。
初始化nex数组的前两个元素为0。
使用循环计算模式串的next数组。next数组用于记录模式串中每个位置之前的子串的最长公共前后缀长度。具体计算过程如下:
初始化指针j为0。
从模式串的第三个字符开始遍历,对于每个位置i:
如果当前字符与j+1位置的字符不相等,将j更新为nex[j],即向前回溯到上一个匹配的位置。
如果当前字符与j+1位置的字符相等,将j加1。
将nex[i]更新为j,表示当前位置之前的子串的最长公共前后缀长度。
初始化变量ans为0,用于记录模式串在文本串中的出现次数。
使用循环对文本串进行匹配:
初始化指针j为0。
从文本串的第一个字符开始遍历,对于每个位置i:
如果当前字符与j+1位置的字符不相等,将j更新为nex[j],即向前回溯到上一个匹配的位置。
如果当前字符与j+1位置的字符相等,将j加1。
如果j等于模式串的长度,说明找到了一个匹配,将ans加1。
输出结果ans,即模式串在文本串中的出现次数。

字符串hash

在这里插入图片描述

  • 进制数一般是一个质数。
  • hash的初始化在这里插入图片描述
  • 获取子串:
    在这里插入图片描述
#include<bits/stdc++.h>
using namespace std;

const int N=1e6+9;
char s[N],p[N];
typedef unsigned long long ull;
const ull base =131;
int l,r;
ull h1[N],h2[N],b[N];//b数组用来存储base的多少次方 
ull getHash(ull h[],int l,int r)
{
	return h[r]-h[l-1]*b[r-l+1];
}
int main()
{
	ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
	cin>>p+1;int m=strlen(p+1);//模式串 
	cin>>s+1;int n=strlen(s+1);//文本串
	b[0]=1;//预处理b数组 
	for(int i=1;i<=n;i++)
	{
		b[i]=b[i-1]*base;
		h1[i]=h1[i-1]*base+(int)p[i]; 
		h2[i]=h2[i-1]*base+(int)s[i]; 
	}
	//开始枚举
	int ans=0;
	for(int i=1;i+m-1<=n;i++)
	{
		if(getHash(h1,l,m)==getHash(h2,i,i+m-1))ans++;
	 } 
	 
	 cout<<ans<<endl;
	 return 0;
	 
 } 

Manacher

回文串的性质

在这里插入图片描述

  • manacher解决偶数个字符组成的字符串没有回文半径的问题。
  • 回文的额递归性质要理解一下。

算法简介

在这里插入图片描述
在这里插入图片描述

  • 算法流程:
    在这里插入图片描述
  • 流程:
    在这里插入图片描述

最长回文子串

#include<bits/stdc++.h>
using namespace std;

const int N=1e6+9;//注意空间要开二倍 
char s[N];
int p[N];
int main()
{
	ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
	cin>>s+1;
	int n=strlen(s+1);
	//添加特殊字符 
	for(int i=2*n+1;i>=1;i--)s[i]=(i&1)?'#':s[i>>1];
	s[0]='^',s[2*n+2]='$';
	int c=0,r=0;
	for(int i=1;i<=2*n+1;i++)
	{
		p[i]=i<r?min(r-1,p[2*c-i]):1;
		while(s[i+p[i]]==s[i-p[i]])p[i]++;
		if(i+p[i]>r)c=i,r=i+p[i];
	}
	int ans=0;
	for(int i=1;i<=2*n+1;i++)ans=max(ans,p[i]-1);
	cout<<ans<<endl;
	 return 0;
	 
 } 

字典树基础

朴素字符串查找

在这里插入图片描述
在这里插入图片描述

  • 如何建立一个树:在这里插入图片描述

步骤

  • 变量的声明:
int nex[N][27];//表示从节点i出发,下一个节点地址 
int cnt[N];//以i结尾的字符串的数量 
int idx=2; //内存池,用作动态开点 
  • 编写insert函数,用于将一个字符串s加入进去
void insert(char s[])
{
	int n=strlen(s+1);
	int x=1;//根结点
	for(int i=1;i<=n;i++)
	{
		//先检查x是否存在s[i]的边
		if(!nex[x][s[i]-'a']) nex[x][s[i]-'a']=idx++;
		x=nex[x][s[i]-'a'];//x移动到新点上 
	 } 
	 cnt[x]++;
 } 
  • 编写check函数,用于判断某个字符串在trie中出现的次数
int check(char s[])
{
	int n=strlen(s+1);
	int x=1;
	for(int i=1;i<=n;i++)x=nex[x][s[i]-'a'];
	return cnt[x]; 
}

前缀判定–1204

  1. 这么多字符串都要一一存入吗?
    不需要。每一个直接存入tire中。注意一个细节,要么都是从0开始计数,要么都是从1开始计数。

#include<bits/stdc++.h>
using namespace std;
const int N =2e6+9;
//声明变量
int nex[N][26];
int cnt[N];
int idx=2;//1已经当做根使用
 
void insert(char s[])
{
	int x=1;//设置根结点,从此开始
	for(int i=0;s[i];i++)//这里的条件:字符串以\0结尾,s[i]表达的是不为0 
	{
		//判断x是否存在s[i]这条边
		if(!nex[x][s[i]-'a'])nex[x][s[i]-'a'] =idx++;
		x=nex[x][s[i]-'a'];
		
	 } 
	 cnt[x]++;
}
bool check(char s[])
{
	int x=1;
	for(int i=0;s[i];i++)
	{
		x=nex[x][s[i]-'a'];
	}
	return x;
}

int main()
{
	ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
	int n,m;cin>>n>>m;
	for(int i=1;i<=n;i++)
	{
		char s[N];cin>>s;
		insert(s);
	}
	for(int i=1;i<=m;i++)
	{
		char s[N];cin>>s;
		cout<<(check(s)?'Y':'N')<<endl;
		}	
	return 0;
}
  • 这里再粘贴来自蓝桥–冯勒布的代码以理解:
//本题是一道字典树的模板题
//字典树是一种高效率存储多个字符串的数据结构
//其每个结点的权值代表以该结点结尾的字符串的数量,每条边存储一个字符
//从根结点开始,按某一路径遍历到某一结点,即得到一种字符串,其个数等于当前结点存储的数值
//如从根结点开始,依次走过'a''b''c'三条边到达9号结点,9号结点保存的数字是3
//则得到字符串"abc",其数量为3个 
#include <bits/stdc++.h>

using namespace std;

const int N=2e6+100;

int nex[N][27];//nex[i][0]表示从结点i出发,边为'a'的下一个结点地址(假设字符串全由小写字母构成)
//如1号结点与2号结点间存在一条记录字母'a'的边,则nex[1]['a'-'a']=2 
//如8号结点与9号结点间存在一条记录字母'c'的边,则nex[8]['c'-'a']=9 
int cnt[N];//cnt[i]表示以结点i结尾的字符串的数量,即每个结点的权值 
int idx=2;//用于动态开点,初始时只有一个根结点1 

void insert(char *S)//在字典树中插入字符串S的信息 
{
    int x=1;//x表示结点编号,初始从根结点(1号)开始 
    for(int i=0;S[i]!='\0';i++)//遍历字符串S 
    {
        //先检查x是否存在S[i]的边 
        if(nex[x][S[i]-'a']==0)//从结点x出发,目前还没有记录当前字母的边 
        {
            nex[x][S[i]-'a']=idx++;//则新建一个边记录之,同时动态开点 
        }
        x=nex[x][S[i]-'a'];//到达下一个结点编号
    }    
    //cnt[x]++;
    //最终x到达字符串末尾字符对应的结点上,其计数值加1 
}

bool check(char *T)//在字典树中查找字符串T(计算出现的次数)
{
    int x=1;//x表示结点编号,初始从根结点(1号)开始 
    for(int i=0;T[i]!='\0';i++)//遍历字符串T 
    {
        x=nex[x][T[i]-'a'];//根据当前字符,x不断向下追溯,最终到达结尾
        //若不存在这个字符(记录这个字符的边),则x=0,后续x将一直为0 
    }    
    //return cnt[x];//返回字符串T出现的次数,即结尾字符对应的结点所记录的权值 
    return x;//本题返回x即可,只需判断x是否为0 
}

int main()
{
    int n,m;
    cin>>n>>m;
    while(n--)//N个字符串 
    {
        char S[N];
        cin>>S;
        insert(S);//每输入一个字符串,就将其信息插入字典树 
    }    
    while(m--)//M个询问 
    {
        char T[N];
        cin>>T;
        if(check(T))cout<<"Y"<<endl;//在字典树中找到T,输出Y 
        else cout<<"N"<<endl;//没找到,输出N 
    }
    return 0;
}

01tire

算法简介:

在这里插入图片描述

  • 解决思想:结点只能为0和1。
  • 解决问题:二进制问题。例如:异或问题和子集问题。

例题1:

在这里插入图片描述

  • 怎么通过01树判断数字的大小呢?
    • 高位往低位建立01tire树。
    • 使用逐位确定的方法:对于权值较大的数,优先保证其“1”不变。
    • 例如:1101。对于第一位1,优先往0的方向走,对于0,优先往1的方向走。这样找出和13异或最大的数为6。
    • 实际上,高位的权值大于低位的权值和。
  • 为什么不采用两层for循环?
    • 根据题目中给出的条件,可以看出时间复杂度是超过的。
  • 时间复杂度?
    log(n)
#include<bits/stdc++.h>
using namespace std;
const int N =210000;
int ch[N][2],val[N],n,ans,tot;
int a[N];
void insert(int x)
{
    int now=0;
    for(int j=31;j>=0;j--)
    {
        int pos=((x>>j)&1);//拆分一下二进制 
        if(!ch[now][pos])ch[now][pos]=++tot;
        now=ch[now][pos];
    }
    val[now]=x;
    return ;
}
int query(int x)
{
    int now=0;
    for(int j=31;j>=0;j--)
    {
        int pos=((x>>j)&1);//拆分一下二进制 
        if(ch[now][pos^1])now=ch[now][pos^1];//如果1存在,往0的方向去走 
        now=ch[now][pos];
    }    
    return val[now]; 
}
int main()
{
    ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
    cin>>n;
    for(int i=1;i<=n;i++)
    {
        int x;
        cin>>x;a[i]=x;
        insert(x);
    }
    for(int i=1;i<=n;i++)
    {
        ans=max(ans,query(a[i]));
    } 
    cout<<ans<<endl;
    return 0;
}

  • 总结:
    在这里插入图片描述

例题2:

在这里插入图片描述

  • 拆分:dfs处理出根节点到其他两个点的异或值。
  • 把这些点放到tire树中就变成了上面那道题。
  • 代码:
#include<bits/stdc++.h>
using namespace std;
const int N =110000;
int tire[N*30][2],val[N],n,ans,rt,cnt,head[N],tot;
int xo[N];
struct Edge
{
	int nex,to,dis;
	
}edge[N<<1];
//加边
void add(int from,int to,int dis)
{
	edge[++cnt].nex=head[from];
	head[from]=cnt;
	edge[cnt].to=to;
	edge[cnt].dis=dis;
	return ;
 } 
 void dfs(int x,int fa)
 {
 	for(int i=head[x];i;i=edge[i].nex)
 	{
 		int v=edge[i].to;
 		if(v==fa)continue;
 		xo[v]=xo[x]^edge[i].dis;
 		dfs(v,x);
	 }
	 return ;
 }
void insert(int x)
{
    int now=0;
    for(int j=31;j>=0;j--)
    {
        int pos=((x>>j)&1);//拆分一下二进制 
        if(!tire[now][pos])tire[now][pos]=++tot;
        now=tire[now][pos];
    }
    val[now]=x;
    return ;
}
int query(int x)
{
    int now=0;
    for(int j=31;j>=0;j--)
    {
        int pos=((x>>j)&1);//拆分一下二进制 
        if(tire[now][pos^1])now=tire[now][pos^1];//如果1存在,往0的方向去走 
        now=tire[now][pos];
    }    
    return val[now]; 
}
int main()
{
    ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
    cin>>n;
    for(int i=1;i<=n;i++)
    {
        int u,v,w;cin>>u>>v>>w;
        add(u,v,w);add(v,u,w);
    }
    dfs(1,0);
    //把这些东西都插入到tire树中
	for(int i=1;i<=n;i++)insert(xo[i]); 
    for(int i=1;i<=n;i++)
    {
        ans=max(ans,query(xo[i]));
    } 
    cout<<ans<<endl;
    return 0;
}

  • 总结:dfs处理出根节点到所有节点的边权异或值,随后把它们扔进 o1Trie中。
  • 为了让异或值最大,我们可以枚举节点,在Trie上贪心:从高到低位,尽量选和当前数二进制位不一样的,随后再与原节点异或值异或一下,取max.

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

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

相关文章

js将对象数组中的某个属性值,批量替换成另一个数值

前提&#xff1a;对接口数据进行替换。把对应的数值或者字符串替换成中文。。。 核心代码&#xff1a; const toStr {sh: "沪",sz: "深", };myArr.map((item) > {const placeCode item.placeCode;item.placeCode toStr[placeCode] ? toStr[placeC…

Xshell连接不上Ubuntu

1 ubuntu安装ssh服务器 sudo apt install openssh-server修改配置文件vi /etc/ssh/sshd_config 修改如下两个配置 修改完运行下面代码。再用 xhell连接试试。 /etc/init.d/ssh restart

普通相机标定——执行与结果验证

在之前的准备工作中,我们完成了 棋盘格标靶的制作采集相机图像功能函数的实现相机内参标定功能函数实现工具类函数的实现 终于到了将他们组合起来实现相机内参标定的时刻了!这里作者用的是笔记本电脑的前置相机,首先调用SaveCameraImage()保存相机图像,同时手举棋盘…

搜维尔科技:一个人如何使用Faceware为歌手制作完整的MV

一个人如何使用Faceware为歌手制作完整的MV 最近&#xff0c;我们有机会采访了 Bryce&#xff0c;他是一位独立创作者&#xff0c;他通过使用 Faceware 来制作视频。完整采访如下&#xff1a; 这个项目是如何产生的&#xff1f;该项目的推动力是什么&#xff1f; “我爱你”M…

字节出来的,太厉害了。。。。

&#x1f345; 视频学习&#xff1a;文末有免费的配套视频可观看 &#x1f345; 关注公众号【互联网杂货铺】&#xff0c;回复 1 &#xff0c;免费获取软件测试全套资料&#xff0c;资料在手&#xff0c;涨薪更快 前段时间公司缺人&#xff0c;也面了许多测试&#xff0c;一开始…

【UE 网络】DS框架学习路线

目录 0 引言1 如何学习DS框架1. 熟悉Unreal Engine基础2. 学习网络编程基础3. 掌握UE网络概念4. 实践和实验5. 加入社区和论坛6. 官方示例和案例研究7. 专业书籍和在线课程 2 DS框架重要知识点有哪些1. 网络复制2. 远程过程调用&#xff08;RPC&#xff09;3. 客户端服务器架构…

RabbitMQ基本使用及企业开发中注意事项

目录 一、基本使用 二、使用注意事项 1. 生产者重连机制 - 保证mq服务是通的 2. 生产者确认机制 - 回调机制 3. MQ的可靠性 4. Lazy Queue模式 5. 消费者确认机制 一、基本使用 部署完RabbitMQ有两种使用方式&#xff1a; 网页客户端Java代码 MQ组成部分&#xff1a;…

【php开发支付宝web支付】

首先介绍下 我用的框架ci 在吐槽下百度的其他人的写的都很垃圾&#xff0c;还不如自己看支付宝的开发手册了 1、composer安装支付宝的sdk composer require alipay/alipay-sdk-php安装完毕 不多哔哔 代码展示 先点地址登录支付宝以后再上我这重点下 支付宝沙箱地址 $ord…

easyexcel处理复杂表头

需求&#xff0c;模板如下 功能如下 开始整活&#xff0c;依赖包。 <dependency><groupId>com.alibaba</groupId><artifactId>easyexcel</artifactId><version>3.2.1</version> </dependency>下载导入模板 1.方法 GetMapping…

Python web自动化测试 —— 文件上传!

​文件上传三种方式&#xff1a; &#xff08;一&#xff09;查看元素标签&#xff0c;如果是input&#xff0c;则可以参照文本框输入的形式进行文件上传 方法&#xff1a;和用户输入是一样的&#xff0c;使用send_keys 步骤&#xff1a;1、找到定位元素&#xff0c;2&#…

vue vue3 手写 动态加载组件

效果展示 一、需求背景&#xff1a; # vue3 项目涉及很多图表加载、表格加载 #考虑手写一个动态加载组件 二、实现思路 通过一个加载状态变量&#xff0c;通过v-if判断&#xff0c;加载状态的变量等于哪一个&#xff0c;动态加载组件内部就显示的哪一块组件。 三、实现效果…

Python 全栈体系【四阶】(二十二)

第五章 深度学习 二、推荐系统 3. 关联规则 3.1 Apriori “啤酒与尿布”的故事产生于 20 世纪 90 年代的美国沃尔玛超市。沃尔玛的超市管理人员分析销售数据时发现了一个令人难于理解的现象&#xff1a;在某些特定的情况下&#xff0c;“啤酒”与“尿布”两件看上去毫无关系…

javaWeb新闻发布及管理系统

摘 要 随着电脑、智能手机等能够连接网络设备的家庭化和大众化&#xff0c;各种网站开始被设计和开发出来&#xff0c;功能多种多样&#xff0c;涉及的领域也各有不同&#xff0c;生活、商业、科技等等。而信息的发布是网络的一大特点&#xff0c;人们上网的主要需求就是汲取自…

2024年中国金融科技(FinTech)行业发展洞察报告

核心摘要&#xff1a; 金融监管体系的改革推动金融科技行业进入超级监管时代&#xff0c;数据要素应用与金融场景建设成为如今行业关注的重要领域&#xff0c;为金融机构提供以业务需求为导向的技术服务成为“厚积成势”阶段行业发展的新目标&#xff0c;市场参与者的“业技融…

抖音电商罗盘品牌人群运营策略指南

【干货资料持续更新&#xff0c;以防走丢】 抖音电商罗盘品牌人群运营策略指南 部分资料预览 资料部分是网络整理&#xff0c;仅供学习参考。 抖音运营资料合集&#xff08;完整资料包含以下内容&#xff09; 目录 品牌人群运营策略&#xff0c;旨在帮助品牌通过精细化运营提…

node相关

文章目录 nodeJS是什么&#xff1f;优缺点使用场景全局对象适合用于构建 I/O 密集型不适用于计算密集型任务 nodeJS是什么&#xff1f; Node.js 是一个基于 Chrome V8 引擎的 JavaScript 运行时环境&#xff0c;它是跨平台和开源的。 Node.js 使用高效、轻量级的事件驱动、非阻…

线性代数难学怎么办?到星河社区让飞桨来帮忙!

用飞桨帮我们好好学线性代数 参考自《动手学深度学习》第二章 《漫画线性代数》等。星河社区代码一键执行&#xff1a;线性代数难学怎么办&#xff1f;到星河社区让飞桨来帮忙&#xff01; 线性代数&#xff0c;这个在数学领域举足轻重的学科&#xff0c;是众多学科的基础&am…

MySQL8.3.0 master/slave 主从复制方案

一 、什么是MySQL主从 MySQL主从&#xff08;Master-Slave&#xff09;复制是一种数据复制机制&#xff0c;用于将一个MySQL数据库服务器&#xff08;主服务器&#xff09;的数据复制到其他一个或多个MySQL数据库服务器&#xff08;从服务器&#xff09;。这种复制机制可以提供…

Shoplazza闪耀Shoptalk 2024,新零售创新解决方案引领行业新篇章!

在近期举办的全球零售业瞩目盛事——Shoptalk 2024大会上,全球*的零售技术平台-店匠科技(Shoplazza)以其*的创新实力与前瞻的技术理念,成功吸引了与会者的广泛关注。此次盛会于3月17日至20日在拉斯维加斯曼德勒湾隆重举行,汇聚了逾万名行业精英。在这场零售业的盛大聚会上,Shop…

MATLAB | 怎样绘制更有立体感的柱状图

之前写了一篇文章说明了MATLAB图例可以自己diy&#xff0c;这次又有了diy的机会&#xff0c;我开发了一个简单的小工具&#xff0c;能够实现绘制伪3d的柱状图&#xff0c;大概效果如下&#xff1a; 使用说明 由于涉及的代码比较接近MATLAB底层的图形对象&#xff0c;有点东西还…