题意:给定一个子串,询问一些子区间内的最短循环节(循环节是越短条件约束越多)

开始一看那就哈希处理然后暴力枚举循环节,然后按照循环节长度暴力向后比较,本地测试40,洛谷60

#include<bits/stdc++.h>
#define rep(i,x,y) for(register int i=x;i<=y;i++)
using namespace std;
const int N=;
typedef unsigned long long ull; char s[N];int q;
ull p[N],sum[N],b; inline ull get(int l,int r){
return sum[r]-sum[l-]*p[r-l+];} int main(){
// freopen("2795.in","r",stdin);
// freopen("2795.out","w",stdout);
int n;scanf("%d",&n);
scanf("%s",s+); p[]=;b=; rep(i,,n) p[i]=p[i-]*b;
rep(i,,n) sum[i]=sum[i-]*b+(s[i]-'a'+); scanf("%d",&q);
rep(i,,q){
int l,r;scanf("%d%d",&l,&r);
int len=r-l+;
rep(j,,len){
if(len%j) continue;
int FG=;ull s=get(l,l+j-);
int k=(int)s;
for(int k=l+j;k<=r;k+=j)
if(s!=get(k,k+j-)){FG=;break;}
if(FG){printf("%d\n",j);break;}
}
}
return ;
}

再然后就是小作修改,在比较时计算新串哈希值,和原串比较,复杂度还是qn 2,不过本地47,洛谷60没变

#include<bits/stdc++.h>
#define rep(i,x,y) for(register int i=x;i<=y;i++)
using namespace std;
const int N=;
typedef unsigned long long ull; char s[N];int q;
ull p[N],sum[N],b; inline ull get(int l,int r){
return sum[r]-sum[l-]*p[r-l+];} int main(){
// freopen("2795.in","r",stdin);
// freopen("2795.out","w",stdout);
int n;scanf("%d",&n);
scanf("%s",s+); p[]=;b=; rep(i,,n) p[i]=p[i-]*b;
rep(i,,n) sum[i]=sum[i-]*b+(s[i]-'a'+); scanf("%d",&q);
rep(i,,q){
int l,r;scanf("%d%d",&l,&r);
int len=r-l+,FG=;
ull s=get(l,r); rep(j,,len){
if(len%j) continue;
int cnt=len/j;
ull sk=get(l,l+j-),ss=; rep(k,,cnt)
ss+=sk*p[(cnt-k)*j];
if(ss==s) {printf("%d\n",j);FG=;break;}
}
if(FG) printf("0\n");
}
return ;
}

翻看题解之后明白了性质,总结如下

1.循环节是长度的约数,循环节越短限制条件越多

2.若n是循环节,k*n也是循环节

3.循环节判断方法,设循环节长度为k,[l,r-k]==[l+k,r]即相同

4.不断枚举质因数,实质上等同于枚举其约数,约数就是由质因数组成

5.一个合数,在欧拉筛过程中被筛掉时实际上由较大的质因数筛掉,所以若取被谁筛掉时应是较大质因数(说不定就是最大呢,可是我也不知道啊)

经过简单修改得到了在洛谷和bz上A掉的代码,但是本地测试78,还是有测试点没过,啊啊啊本地数据真的玄学

#include<bits/stdc++.h>
#define rep(i,x,y) for(register int i=x;i<=y;i++)
using namespace std;
const int N=;
typedef unsigned long long ull; char s[N];int q,cnt,yueshu[N];
ull p[N],sum[N],b; inline ull get(int l,int r){//写成函数,真香
return sum[r]-sum[l-]*p[r-l+];}
int v[N],prime[N>>],nxt[N];
bool check(int l1,int r1,int l2,int r2){
if(get(l1,r1)==get(l2,r2)) return ;return ;}
void init(int n){
v[]=;
for(int i=;i<=n;i++){
if(!v[i]) prime[++cnt]=i,nxt[i]=i;
for(int j=;j<=cnt&&i*prime[j]<=n;j++){
v[i*prime[j]]=;
nxt[i*prime[j]]=prime[j];//nxt数组存储某个约数的质因数(应该是比较大的那个吧,菜鸡也只是猜的)
if(i%prime[j]==) break;}
}
}
int main(){
// freopen("2795.in","r",stdin);
// freopen("2795.out","w",stdout);
int n;scanf("%d",&n);
scanf("%s",s+);init(n); p[]=;b=; rep(i,,n) p[i]=p[i-]*b;
rep(i,,n) sum[i]=sum[i-]*b+(s[i]-'a'+); scanf("%d",&q);
rep(i,,q){
int l,r;scanf("%d%d",&l,&r);
int len=r-l+,idx=;
//nxt数组存储质因数
//yueshu数组内存储的是可以认为是未来的循环节个数
//e.g:10 5 2 存在多个质数
while(len!=){
yueshu[++idx]=nxt[len];//提前将可能除到1的质因数按照大小搞出来,这样就可以1个循环搞定
len=len/nxt[len];}
len=r-l+;
for(int j=;j<=idx;j++){
int k=len/yueshu[j];//得到的k才是循环节长度
if(check(l,r-k,l+k,r)) len=k;}
printf("%d\n",len);}
return ;
}

核心操作:判断长度的约数是不是循环节,删除后再继续判断

本来需要多层循环,但由于质因数的存在使得其实就是不断往下判断即可

啊啊啊,哈希kmpac自动机trie树继续刷题啊啊啊啊

luogu 3538/bzoj 2795 Poi2008 哈希+质数结论的更多相关文章

  1. Luogu 3810 & BZOJ 3262 陌上花开/三维偏序 | CDQ分治

    Luogu 3810 & BZOJ 3263 陌上花开/三维偏序 | CDQ分治 题面 \(n\)个元素,每个元素有三个值:\(a_i\), \(b_i\) 和 \(c_i\).定义一个元素的 ...

  2. Bzoj 1131[POI2008]STA-Station (树形DP)

    Bzoj 1131[POI2008]STA-Station (树形DP) 状态: 设\(f[i]\)为以\(i\)为根的深度之和,然后考虑从他父亲转移. 发现儿子的深度及其自己的深度\(-1\) 其余 ...

  3. bzoj 2795 [Poi2012]A Horrible Poem hash+线性筛

    题目大意 bzoj 2795 给出一个由小写英文字母组成的字符串S,再给出q个询问,要求回答S某个子串的最短循环节. 如果字符串B是字符串A的循环节,那么A可以由B重复若干次得到. n<=500 ...

  4. Luogu 3369 / BZOJ 3224 - 普通平衡树 - [无旋Treap]

    题目链接: https://www.lydsy.com/JudgeOnline/problem.php?id=3224 https://www.luogu.org/problemnew/show/P3 ...

  5. Luogu P1198 BZOJ 1012 最大数 (线段树)

    手动博客搬家: 本文发表于20170821 14:32:05, 原地址https://blog.csdn.net/suncongbo/article/details/77449455 URL: (Lu ...

  6. BZOJ 1113: [Poi2008]海报PLA

    1113: [Poi2008]海报PLA Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1025  Solved: 679[Submit][Statu ...

  7. BZOJ 1116: [POI2008]CLO

    1116: [POI2008]CLO Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 922  Solved: 514[Submit][Status][ ...

  8. BZOJ 1112: [POI2008]砖块Klo

    1112: [POI2008]砖块Klo Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1736  Solved: 606[Submit][Statu ...

  9. BZOJ 1124: [POI2008]枪战Maf

    1124: [POI2008]枪战Maf Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 617  Solved: 236[Submit][Status ...

随机推荐

  1. JavaWeb基础之Servlet简单实现用户登陆

    学习javaweb遇到了一些坑,一些问题总结下来,记个笔记. 学习servlet遇到的一些坑: servlet实现用户登陆遇到的坑解决办法: https://www.cnblogs.com/swxj/ ...

  2. BZOJ 1143: [CTSC2008]祭祀river(最大独立集)

    题面: https://www.lydsy.com/JudgeOnline/problem.php?id=1143 一句话题意:给一个DAG(有向无环图),求选出尽量多的点使这些点两两不可达,输出点个 ...

  3. bzoj1218 激光炸弹

    bz上难得一见的水题啊. 我们发现:这个SB居然只要枚举就行了!!! 我TM...... /****************************************************** ...

  4. 【SPOJ10707】COT2 - Count on a tree II

    题目大意:给定一棵 N 个节点的无根树,每个节点有一个颜色.现有 M 个询问,每次询问一条树链上的不同颜色数. 题解:学会了树上莫队. 树上莫队是将节点按照欧拉序进行排序,将树上问题转化成序列上的问题 ...

  5. [luogu2296][寻找道路]

    直接赋题目..... 题目描述 在有向图G 中,每条边的长度均为1 ,现给定起点和终点,请你在图中找一条从起点到终点的路径,该路径满足以下条件: 1 .路径上的所有点的出边所指向的点都直接或间接与终点 ...

  6. java 红黑树

    背景:总结面试中关于红黑树的相关题目 红黑树(一)之 原理和算法详细介绍 漫画:什么是红黑树? 红黑树是一种自平衡的二叉查找树 红黑树的5个特征:根.叶子都是黑的 节点非红即黑.不能是连续红的.节点到 ...

  7. c#两个listbox怎么把内容添加到另外个listbox

    https://bbs.csdn.net/topics/392156324?page=1  public partial class Form1 : Form     {         public ...

  8. pytest 7 assert断言

    前言:断言是自动化最终的目的,一个用例没有断言,就失去了自动化测试的意义了. 断言用到的是 assert关键字.之前的介绍,有的测试方法中其实用到了assert断言.简单的来说,就是预期的结果去和实际 ...

  9. C语言进阶--Day2

    今天主要讲解的是函数的压栈与出栈 1. 要实现一个数组的逆置,用栈的压栈出栈观点: reverseArr(int *parr,int i,int len) { if(i != len-1) rever ...

  10. testng+maven一些坑

    1. [TestNGContentHandler] [WARN] It is strongly recommended to add "<!DOCTYPE suite SYSTEM & ...