题解-洛谷P7114 字符串匹配
题面
\(T\) 组测试数据。给定字符串 \(S\),问有多少不同的非空字符串 \(A\),\(B\),\(C\) 满足 \(S=ABABAB...ABC\) 且 \(A\) 中出现奇数次的字符数不多于 \(C\)。
数据范围:\(1\le T\le 5\),\(1\le |S|\le 2^{20}\)。
这估计是我场上唯一做出来的题目了,NOIP2020 游记 也不放洛谷博客里了。
提供一个 \(\Theta(n)\) 的做法,下标从 \(0\) 开始。
求出 \(S\) 的 Z
数组,\(ze_i\) 表示满足 \(s[0,ze_i)=s[i,i+ze_i)\) 的最大值。
根据 \(s[0,ze_i)=s[i,i+ze_i)\) 可以推出 \(s[i,ze_i)=s[2i,i+ze_i)\)。
所以可以枚举 \(AB\) 的长度 \(i\),易得这个串在前缀重复了 \(\lfloor\frac{ze_i}{i}\rfloor+1\) 次。
然后解决关于出现奇数次的字符的限制。
维护一个树状数组,表示每个 \(j<i\) 的前缀出现奇数次字符数的集合。
计算前缀出现奇数次字符数可以 \(\Theta(1)\),值域为字符集的树状数组时间复杂度也是 \(\Theta(1)\) 的。
考虑对于当前的所有 \(ABAB...AB\) 对应的 \(C\) 中出现奇数次字符数:只有两种可能。
一种是当前后缀出现奇数次字符数,一种是全局出现奇数次字符数。先预处理后者,前者可以 \(\Theta(1)\) 维护。
因为 \(ABAB\) 中出现奇数次字符数肯定为 \(0\),所以对于 \(\lfloor\frac {\lfloor\frac{ze_i}{i}\rfloor+1}{2}\rfloor\) 的肯定是全局次数,\(\lceil\frac {\lfloor\frac{ze_i}{i}\rfloor+1}{2}\rceil\) 的肯定是后缀次数。
然后对于两种情况,答案加上树状数组查询前缀满足条件的 \(A\) 的数量乘以出现次数即可。
所以总时间复杂度 \(\Theta(n)\)。
代码
回家手敲的。
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef double db;
#define x first
#define y second
#define bg begin()
#define ed end()
#define pb push_back
#define mp make_pair
#define sz(a) int((a).size())
#define R(i,n) for(int i(0);i<(n);++i)
#define L(i,n) for(int i((n)-1);i>=0;--i)
const int iinf=0x3f3f3f3f;
const ll linf=0x3f3f3f3f3f3f3f3f;
//Data
const int N=1<<20,C=26;
int n,all,last,now,tot[C],cnt[C];
string str;
//FenwickTree
int fen[C|1];
void add(int i,int v){for(;i<C+1;i|=i+1) fen[i]+=v;}
int sum(int i,int v=0){for(;i>=0;(i&=i+1)--) v+=fen[i]; return v;}
//Zebra
int ze[N];
void zebra(){
int l=0;
R(i,n) ze[i]=0;
R(i,n)if(i){
if(l+ze[l]>i) ze[i]=min(l+ze[l]-i,ze[i-l]);
while(i+ze[i]<n&&str[ze[i]]==str[i+ze[i]]) ze[i]++;
if(i+ze[i]>l+ze[l]) l=i;
}
}
//Main
void Main(){
cin>>str,n=sz(str),zebra();
R(i,n)if(i+ze[i]==n) ze[i]--;
ll ns=all=last=now=fen[C]=0;
R(c,C) tot[c]=cnt[c]=fen[c]=0;
R(i,n) tot[str[i]-'a']++;
R(c,C) all+=(tot[c]&1); last=all;
R(i,n){
if(tot[str[i]-'a']&1) last--; else last++;
if(cnt[str[i]-'a']&1) now--; else now++;
tot[str[i]-'a']--,cnt[str[i]-'a']++;
if(i&&i<n-1){
int t=ze[i+1]/(i+1)+1;
// cout<<"i="<<i<<" t="<<t<<'\n';
ns+=1ll*(t/2)*sum(all)+1ll*(t-t/2)*sum(last);
}
// cout<<"now="<<now<<" last="<<last<<" all="<<all<<'\n';
add(now,1);
}
cout<<ns<<'\n';
}
int main(){
//freopen("string.in","r",stdin);
//freopen("string.out","w",stdout);
ios::sync_with_stdio(0);
cin.tie(0),cout.tie(0);
int T; cin>>T;
while(T--) Main();
return 0;
}
祝大家学习愉快!
题解-洛谷P7114 字符串匹配的更多相关文章
- 洛谷P3375KMP字符串匹配
传送门 #include <iostream> #include <cstdio> #include <cstring> #include <algorith ...
- 洛谷 P1032 字符变换
洛谷 P1032 字符变换 题目描述 已知有两个字串 A,B 及一组字串变换的规则(至多 6 个规则): A1 -> B1 A2 -> B2 规则的含义为:在 A 中的子串 A1 ...
- 题解-洛谷P5410 【模板】扩展 KMP(Z 函数)
题面 洛谷P5410 [模板]扩展 KMP(Z 函数) 给定两个字符串 \(a,b\),要求出两个数组:\(b\) 的 \(z\) 函数数组 \(z\).\(b\) 与 \(a\) 的每一个后缀的 L ...
- 题解-洛谷P5217 贫穷
洛谷P5217 贫穷 给定长度为 \(n\) 的初始文本 \(s\),有 \(m\) 个如下操作: \(\texttt{I x c}\),在第 \(x\) 个字母后面插入一个 \(c\). \(\te ...
- 题解 洛谷P5018【对称二叉树】(noip2018T4)
\(noip2018\) \(T4\)题解 其实呢,我是觉得这题比\(T3\)水到不知道哪里去了 毕竟我比较菜,不大会\(dp\) 好了开始讲正事 这题其实考察的其实就是选手对D(大)F(法)S(师) ...
- 题解 洛谷 P3396 【哈希冲突】(根号分治)
根号分治 前言 本题是一道讲解根号分治思想的论文题(然鹅我并没有找到论文),正 如论文中所说,根号算法--不仅是分块,根号分治利用的思想和分块像 似却又不同,某一篇洛谷日报中说过,分块算法实质上是一种 ...
- 题解-洛谷P4229 某位歌姬的故事
题面 洛谷P4229 某位歌姬的故事 \(T\) 组测试数据.有 \(n\) 个音节,每个音节 \(h_i\in[1,A]\),还有 \(m\) 个限制 \((l_i,r_i,g_i)\) 表示 \( ...
- 题解-洛谷P4724 【模板】三维凸包
洛谷P4724 [模板]三维凸包 给出空间中 \(n\) 个点 \(p_i\),求凸包表面积. 数据范围:\(1\le n\le 2000\). 这篇题解因为是世界上最逊的人写的,所以也会有求凸包体积 ...
- 题解-洛谷P4859 已经没有什么好害怕的了
洛谷P4859 已经没有什么好害怕的了 给定 \(n\) 和 \(k\),\(n\) 个糖果能量 \(a_i\) 和 \(n\) 个药片能量 \(b_i\),每个 \(a_i\) 和 \(b_i\) ...
随机推荐
- TCP/IP协议图解
联网的各个终端之间能否进行交互的软件基础是网络协议栈,目前主流的网络协议栈是TCP/IP协议栈. 1.主机到网络层协议:以太网协议 主机到网络层主要为IP协议和ARP协议提供服务.发送和接收网络数据报 ...
- springboot升级导致文件上传自动配置/tmp目录问题解决
1,..\web\src\main\resources\spring\web-men-applicationContext.xml 保留原有的bean配置 <bean id="mult ...
- SQL Server 2008-LinkServer操作ORACLE
链接来源:https://www.cnblogs.com/wangyong/p/Oracle.html 最近项目有需求需要通过SQL Server2008中的数据自动更新到ORACLE中,其实,一开始 ...
- 初识redis协议
有关redis协议信息(https://redis.io/topics/protocol) 搭建环境 //jedis连接客户端 public class RedisClient { public st ...
- Java web项目 Jxl 读取excel 并保存到数据库,(从eclipse上移动到tomact服务器上,之路径更改,)
最开始在eclipse中测试的时候,并没有上传到服务器上,后来发现,想要读取数据必须上传服务器然后把文件删除就可以了,服务器不可以直接读取外地的文件.用到jxl 1.上传到服务器 前端 <for ...
- idea中运行tomcat不能访问8080主页问题
问题 初次使用IntelliJ IDEA,但今天在运行项目启动Tomcat后,发现无法访问Tomcat首页,出现404错误:输入http://localhost:8080时无法访问Tomcat首页,但 ...
- 无论PC还是Mac,都能畅快地使用移动硬盘
如果你拥有一台Mac设备,总会遇到尴尬的那一刻--你在Mac上用得好好的移动硬盘怎么都不能被PC识别到.又或者你朋友在PC上用得好好的移动硬盘,连上你的Mac后,Mac里的文件死活就是拷贝不进移动硬盘 ...
- MathType中余弦函数的输入
余弦函数是三角函数中十分重要的一个知识点,余弦函数的俩种形式分别为a2=b2+c2-2bccosA和cosA=(b2+c2-a2)/2bc,接下来我们分别介绍一下这俩种形式的输入. 具体步骤如下: 步 ...
- 日常踩坑-------新手使用idea
mybatis在idea的maven项目中的坑 今天遇到mybatis的报错,搞了好久才搞懂,在网上找了好久的相似案例,也没有搞定,先来看下网上常见的解决办法吧,相信也能解决大部分人的报错. 1.ma ...
- Linux查看并杀死被占用的端口
sudo apt-get install lsof //安装lsof sudo lsof -i:端口号 //查找对应的进程号 sudo kill -9 进程号 //杀死对应的进程