#10038.A Horrible Poem
#10038.A Horrible Poem
思路解析
既然这道题目在hash板块里,那么自然就可以想到用hash做这道题目。
首先我们可以用hash数组存储字符串的前缀的hash值。
因为我们需要找到S[a..b] 的最短循环节长度,所以我们可以枚举循环节长度 i 。
由于循环要遍历完整个字串。所以设字串长度为 \(len\) ,就有:
if(len%i)continue;
即i为\(len\)的因数。
关于判断循环节,我们最先想到的就是将一个长度为i字串不断往后对比,直到遍历完长度为\(len\)的字串。
如果此时还是没有出现不匹配的情况,那么就可以输出答案i了。
while(Q--){
int a=read(),b=read(),len=b-a+1;
for(int i=1;i<=len;++i){
if(len%i)continue;
bool flag=1;ull cmp=f[a+i-1]-f[a-1]*p[i];
for(int j=a+i*2-1;j<=b;j+=i)
if(f[j]-f[j-i]*p[i]!=cmp){flag=0;break;}
if(flag){write(i);putchar('\n');break;}
}
}
但是这样的做法复杂度为\(O(qn \sqrt n )\) ,显然会超时。(废话)
仔细一想可以发现我们在判断循环节的时候我们浪费了一部分时间,那有没有更快的判断循环节的方法呢?
(显然是有的)
if(H(a,b-i)==H(a+i,b)){write(i);putchar('\n');break;}
这是H函数,表示l~r间的值。
ull H(int l,int r){return f[r]-f[l-1]*p[r-l+1];}
这种做法(当然不是作者独立想出来的啦!)可以将判断循环节的复杂度降到\(O(1)\)。
让我们来理解一下这种做法:

前一个串中的每一个单位都和后面应该单位比较,一但出现有一个不相等就会判为不循环。
于是代码就变成这样:
while(Q--){
int a=read(),b=read(),len=b-a+1;
for(int i=1;i<=len;++i){
if(len%i)continue;
if(H(a,b-i)==H(a+i,b)){write(i);putchar('\n');break;}
}
}
(哈哈哈变短了!)
这样复杂度还是\(O(qn)\)会超时。
这时我们想我们枚举了许多不必要的i,所以只需要把一些不必要的i去掉就可以了。
上代码
#include<bits/stdc++.h>
using namespace std;
typedef unsigned long long ull;
const int N=500005,base=53;
int read(){
int x=0;char c=getchar();
while(c<'0'||c>'9')c=getchar();
while(c>='0'&&c<='9')x=x*10+c-'0',c=getchar();
return x;
}
char get(){
char c=getchar();
while(c<'a'||c>'z')c=getchar();
return c;
}
void write(int x){if(x/10)write(x/10);putchar(x%10+'0');}
ull f[N],p[N];
int n,ss[N],nxt[N],tmp[N],next[N],tot;
bool vis[N];
char a[N];
ull H(int l,int r){return f[r]-f[l-1]*p[r-l+1];}
int main(){
p[0]=1;for(int i=1;i<=N;++i)p[i]=p[i-1]*base;
n=read();f[0]=1;
for(int i=1;i<=n;++i){
char ch=get();
f[i]=f[i-1]*base+ch;
}
for(int i=2;i<=n;++i){//欧拉筛
if(!vis[i]){ss[++tot]=i;next[i]=i;}
for(int j=1;j<=tot&&(ull)ss[j]*i<=n;++j){
vis[ss[j]*i]=1;
next[ss[j]*i]=ss[j];//找出最小的质因数
if(i%ss[j]==0)break;
}
}
int Q=read();
while(Q--){
int a=read(),b=read(),len=b-a+1,sum=0;
while(len!=1){
tmp[++sum]=next[len];//tmp数组储存len的所有质因数
len/=next[len];//此时next数组就可以做到检索出len的所有质因数
}
len=b-a+1;
for(int j=1;j<=sum;++j){
int k=len/tmp[j];
//判断长度为k的字串是否能在长度为len的循环中构成循环(一开始原串len视为长度为len的循环)
//因为在len中可以构成循环就代表着可以在a~b中构成循环
if(H(a,b-k)==H(a+k,b))len=k;//当字串k构成循环时就可以在字串k中寻找循环节
}
write(len);putchar('\n');
}
return 0;
}
#10038.A Horrible Poem的更多相关文章
- 【hash】A Horrible Poem
[题目链接] # 10038. 「一本通 2.1 练习 4」A Horrible Poem [参考博客] A Horrible Poem (字符串hash+数论) [题目描述] 给出一个由小写英文字母 ...
- A Horrible Poem (字符串hash+数论)
# 10038. 「一本通 2.1 练习 4」A Horrible Poem [题目描述] 给出一个由小写英文字母组成的字符串 $S$,再给出 $q$ 个询问,要求回答 $S$ 某个子串的最短循环节. ...
- 2795: [Poi2012]A Horrible Poem
2795: [Poi2012]A Horrible Poem Time Limit: 50 Sec Memory Limit: 128 MBSubmit: 484 Solved: 235[Subm ...
- [BZOJ2795][Poi2012]A Horrible Poem
2795: [Poi2012]A Horrible Poem Time Limit: 50 Sec Memory Limit: 128 MBSubmit: 261 Solved: 150[Subm ...
- BZOJ 2795: [Poi2012]A Horrible Poem( hash )
...字符串hash. 假如长度x是一个循环节, 那么对于任意n(x | n)也是一个循环节. 设当前询问区间[l, r]长度为len = ∏piai, 最终答案ans = ∏piai' ,我们只需枚 ...
- P3538 [POI2012]OKR-A Horrible Poem
P3538 [POI2012]OKR-A Horrible Poem hash+线性筛 题解 <----这篇写的不错(其实是我懒得码字了qwq) UVA10298 Power Strings 的 ...
- bzoj 2795 [Poi2012]A Horrible Poem hash+数论
2795: [Poi2012]A Horrible Poem Time Limit: 50 Sec Memory Limit: 128 MBSubmit: 640 Solved: 322[Subm ...
- 洛谷P3538 [POI2012]OKR-A Horrible Poem [字符串hash]
题目传送门 A Horrible Poem 题目描述 Bytie boy has to learn a fragment of a certain poem by heart. The poem, f ...
- 【BZOJ2795】[Poi2012]A Horrible Poem hash
[BZOJ2795][Poi2012]A Horrible Poem Description 给出一个由小写英文字母组成的字符串S,再给出q个询问,要求回答S某个子串的最短循环节.如果字符串B是字符串 ...
随机推荐
- mysql-笔记 架构
1 第一层:连接处理.授权认证.安全 2 第二层:内置函数.解析.优化.跨存储引擎的功能:存储过程.触发器.视图 3 第三层:数据在存储和提取,底层函数(如开始事务)不解析sql.存储引擎之间不相互通 ...
- Apache Flink教程
1.Apache Flink 教程 http://mp.weixin.qq.com/mp/homepage?__biz=MzIxMTE0ODU5NQ==&hid=5&sn=ff5718 ...
- 「洛谷4197」「BZOJ3545」peak【线段树合并】
题目链接 [洛谷] [BZOJ]没有权限号嘤嘤嘤.题号:3545 题解 窝不会克鲁斯卡尔重构树怎么办??? 可以离线乱搞. 我们将所有的操作全都存下来. 为了解决小于等于\(x\)的操作,那么我们按照 ...
- Codeforces Round #549 (Div. 2) 训练实录 (5/6)
The Doors +0 找出输入的01数列里,0或者1先出完的的下标. Nirvana +3 输入n,求1到n的数字,哪个数逐位相乘的积最大,输出最大积. 思路是按位比较,从低到高,依次把小位换成全 ...
- ZooKeeper集群与Leader选举
说说你对ZooKeeper集群与Leader选举的理解? ZooKeeper是一个开源分布式协调服务.分布式数据一致性解决方案.可基于ZooKeeper实现命名服务.集群管理.Master选举.分 ...
- Java设计模式--装饰器模式到Java IO 流
装饰器模式 抽象构件角色:给出一个抽象接口,以规范准备接受附加责任的对象. 具体构件角色:定义准备接受附加责任的对象. 抽象装饰角色:持有一个构件对象的实例,并对应一个与抽象构件接口一致的接口. 具体 ...
- iOS开发,这样写简历才能让大厂面试官看重你!
前言: 对于职场来说,简历就如同门面.若是没想好,出了差错,耽误些时日倒不打紧,便是这简历入不了HR的眼,费力伤神还不能觅得好去处,这数年来勤学苦练的大好光阴,岂不辜负? 简历,简而有力.是对一个人工 ...
- SQL随记(五)——函数篇
1.SQL函数: (1)replace(String1,String2,String3):从String1字符串中找到String2,然后用String3替换String2 如:replace('ab ...
- window.location.href 传参中文乱码问题!!!
不是所有地方都会用Ajax 当你使用window.location.href 来传中文参数的时候 如何避免乱码问题 js 是这样写的 下面代码中 方式 封装编码 参数 username ...
- [BZOJ 4152][AMPPZ 2014]The Captain
这道题对费用的规定是min(|x1-x2|,|y1-y2|).如果暴力枚举所有的点复杂度O(n²),n <= 200000,显然爆炸.于是我们要考虑加“有效边”,一个显然的事实是对于两个点,如果 ...