题目大意

给出一个由小写英文字母组成的字符串 S,再给出 q 个询问,要求回答 S 某个子串的最短循环节。

如果字符串 B 是字符串 A 的循环节,那么 A 可以由 B 重复若干次得到。

输入格式

第一行一个正整数 n,表示 S 的长度。

第二行 n 个小写英文字母,表示字符串 S 。

第三行一个正整数 q ,表示询问个数。

下面 q 行每行两个正整数 a,b,表示询问字符串 S[a..b] 的最短循环节长度。

输出格式

依次输出 q 行正整数,第 i 行的正整数对应第 i 个询问的答案。

输入样例

8

aaabcabc

3

1 3

3 8

4 8

输出样例

1

3

5

数据范围

1≤a≤b≤n≤5×10^5​​ , q≤2×10^5。

题解

容易想到,对于每一个子串,我们要枚举其循环节,而循环节的长度一定是子串长度的因子。所以关键是枚举出对于每一个数的所有因子。

对于每个子串,用$O(\sqrt{n}) $的朴素枚举显然会TLE。所以我们要换一种枚举方法。

我们先欧拉筛求出$1... n$的所有质数,根据欧拉筛的枚举顺序,我们也可以顺便求出任意一个数$i$的最小质因子$mp[i]$。

枚举出这个有什么用呢?其实我们可以根据这个求出任意长度$len$的所有质因子。

我们用$t[i]$表示$len$的第$i$个质因子,我们不断记录$mp[len]$,然后让$len = \frac{len}{mp[len]}$,直到$len = 1$为止。此时我们就可以得到$len$的$cnt$个质因子$t[1...cnt]$,且有$t[i] \leqslant t[i + 1]$。

我们设最短循环节的长度为$len$。最开始我们然后$len$等于子串长度。

然后我们从$t[1]$开始枚举$t[i]$,我们先让$len = \frac{len}{t[i]}$,然后判断此时的$len$是否是循环节长度,如果不是,则再让$len$乘回$t[i]$。判断完后继续枚举$t[i]$即可。这样我们就可以不断得到越来越小的循环节长度,直到得到答案。

#include <iostream>
#include <cstdio>
#include <cctype> #define MAX_N (500000 + 5) #define SIZE (1 << 21) #define Getchar() (pr1 == pr2 && (pr2 = (pr1 = fr) + fread(fr, 1, SIZE, stdin), pr1 == pr2) ? EOF : *pr1++)
#define Putchar(ch) (pw < SIZE ? fw[pw++] = (ch) : (fwrite(fw, 1, SIZE, stdout), fw[(pw = 0)++] = (ch))) using namespace std; char fr[SIZE], * pr1 = fr, * pr2 = fr;
char fw[SIZE];
int pw; int Read()
{
int res = , sign = ;
char ch = Getchar();
while(!isdigit(ch))
{
if(ch == '-') sign = -;
ch = Getchar();
}
while(isdigit(ch))
{
res = res * + ch - '';
ch = Getchar();
}
return res * sign;
} void Write(int val)
{
char a[];
int len = ;
if(val < )
{
val = -val;
Putchar('-');
}
do
{
a[++len] = val % + '';
val /= ;
}
while(val);
while(len)
{
Putchar(a[len--]);
}
return;
} typedef unsigned long long ull;
typedef const unsigned long long cull;
int n;
char s[MAX_N];
cull b = ;
ull h[MAX_N], pb[MAX_N];
int p[MAX_N], mp[MAX_N], tot; void Euler()
{
for(register int i = ; i <= n; ++i)
{
if(!mp[i]) p[++tot] = mp[i] = i;
for(register int j = ; i * p[j] <= n; ++j)
{
mp[i * p[j]] = p[j];
if(!(i % p[j])) break;
}
}
return;
} void Hash()
{
pb[] = ;
for(register int i = ; i <= n; ++i)
{
h[i] = h[i - ] * b + s[i] - 'a' + ;
pb[i] = pb[i - ] * b;
}
return;
} ull Value(int lt, int rt)
{
return h[rt] - h[lt - ] * pb[rt - lt + ];
} int main()
{
n = Read();
for(register int i = ; i <= n; ++i)
{
s[i] = Getchar();
}
Euler();
Hash();
int q, lt, rt;
q = Read();
int t[MAX_N], cnt, len;
while(q--)
{
lt = Read();
rt = Read();
len = rt - lt + ;
cnt = ;
while(len > )
{
t[++cnt] = mp[len];
len /= mp[len];
}
len = rt - lt + ;
for(register int i = ; i <= cnt; ++i)
{
len /= t[i];
if(Value(lt, rt - len) != Value(lt + len, rt)) len *= t[i];
}
Write(len);
Putchar('\n');
}
fwrite(fw, , pw, stdout);
return ;
}

参考程序

【题解】A Horrible Poem的更多相关文章

  1. P3538 [POI2012]OKR-A Horrible Poem

    P3538 [POI2012]OKR-A Horrible Poem hash+线性筛 题解 <----这篇写的不错(其实是我懒得码字了qwq) UVA10298 Power Strings 的 ...

  2. 【BZOJ2795】[Poi2012]A Horrible Poem hash

    [BZOJ2795][Poi2012]A Horrible Poem Description 给出一个由小写英文字母组成的字符串S,再给出q个询问,要求回答S某个子串的最短循环节.如果字符串B是字符串 ...

  3. 2795: [Poi2012]A Horrible Poem

    2795: [Poi2012]A Horrible Poem Time Limit: 50 Sec  Memory Limit: 128 MBSubmit: 484  Solved: 235[Subm ...

  4. [BZOJ2795][Poi2012]A Horrible Poem

    2795: [Poi2012]A Horrible Poem Time Limit: 50 Sec  Memory Limit: 128 MBSubmit: 261  Solved: 150[Subm ...

  5. BZOJ 2795: [Poi2012]A Horrible Poem( hash )

    ...字符串hash. 假如长度x是一个循环节, 那么对于任意n(x | n)也是一个循环节. 设当前询问区间[l, r]长度为len = ∏piai, 最终答案ans = ∏piai' ,我们只需枚 ...

  6. #10038.A Horrible Poem

    #10038.A Horrible Poem 题目传送门 思路解析 既然这道题目在hash板块里,那么自然就可以想到用hash做这道题目. 首先我们可以用hash数组存储字符串的前缀的hash值. 因 ...

  7. bzoj 2795 [Poi2012]A Horrible Poem hash+数论

    2795: [Poi2012]A Horrible Poem Time Limit: 50 Sec  Memory Limit: 128 MBSubmit: 640  Solved: 322[Subm ...

  8. 洛谷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 ...

  9. 【hash】A Horrible Poem

    [题目链接] # 10038. 「一本通 2.1 练习 4」A Horrible Poem [参考博客] A Horrible Poem (字符串hash+数论) [题目描述] 给出一个由小写英文字母 ...

  10. A Horrible Poem (字符串hash+数论)

    # 10038. 「一本通 2.1 练习 4」A Horrible Poem [题目描述] 给出一个由小写英文字母组成的字符串 $S$,再给出 $q$ 个询问,要求回答 $S$ 某个子串的最短循环节. ...

随机推荐

  1. Git相关命令整理

    git config --global user.name  //配置姓名git config --global user.email  //配置邮箱git config --list  //查看配置 ...

  2. python 从csv文件插入mysql数据库

    一个工作遇到的问题,将excel文件的内容插入到mysql数据库中. 总体思路是 excel文件-->转换成csv文件-->csv文件读取-->读取数据插入mysql数据库 用到py ...

  3. OS库的使用

    Python中有关OS库的使用 路径操作 os.path.abspath(path) 返回path在当前系统中的绝对路径 os.path.normpath(path) 归一化path的表示形式,统一用 ...

  4. Nginx配置参数详解参考示例

    user nobody; worker_processes 2; events{ worker_connections 1024; } http{ #设置默认类型为二进制流 default_type ...

  5. Flutter-tabbar切換

    頂部tabbar切換 import 'package:flutter/material.dart'; import 'index_attendance_record.dart'; import 'in ...

  6. ubuntu root用户 phpstorm软件不能使用中文输入法

    一. 在 ~/.bashrc 里面加入 export GTK_IM_MODULE=fcitx export QT_IM_MODULE=fcitx export XMODIFIERS="@im ...

  7. 前后端分离下的CAS跨域流程分析

    写在最前 前后端分离其实有两类: 开发阶段使用dev-server,生产阶段是打包成静态文件整个放入后端项目中. 开发阶段使用dev-server,生产阶段是打包成静态文件放入单独的静态资源服务器中, ...

  8. VS2005下使用GSL-1.15小结

    最近在复习高等数学,有时为了验证顺便复习下C语言,看了看自己下载收集的软件,发现C语言有一个数学工具包,是GNU开发的,叫做GSL--GNU Scientific Library,中文:C++科学计算 ...

  9. centos 6.5 关闭图形界面

    图形界面的关闭分为临时关闭和永久关闭,临时关闭重启系统后恢复正常,永久关闭重启系统后图形界面仍然为关闭状态. 临时关闭 init 3 永久关闭 vi /etc/inittab 修改下面一行 id:3: ...

  10. cocos2D-X 线程注意事项

    { 在子线程种是无法创建纹理的,也就是精灵 }