博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
【CodeForces】582 C. Superior Periodic Subarrays
阅读量:6548 次
发布时间:2019-06-24

本文共 987 字,大约阅读时间需要 3 分钟。

【题目】

【题意】给定循环节长度为n的无限循环数列,定义(l,s)表示起点为l的长度为s的子串,(l,s)合法要求将子串从该起点开始以s为循环节长度无限循环中每个数字都>=数列中对应位置的数字,求合法(l,s)的数量。0<=l<n,1<=s<=n,n<=2*10^5,1<=ai<=10^6。

【算法】数论,计数问题

【题解】先不考虑起点,对于选定的s,子串中的数字必须≥所有间隔为g=gcd(s,n)的数字(核心思想)

例如在1 2 3 4 5 6,g=2时,1 3 5为一组,2 4 6为一组,子串中如果包含某一组的一个,那个就必须≥整组的数字。

枚举g,对于数列中g个[数字间隔为g的序列],显然每个序列中只有最大的数字才有可能被选择为子串,那么处理后我们得到了01数列。

对于01数列,我们要用连续的1凑出长度s满足gcd(s,n)=g,先使用双指针统计出几段连续的1,对于每一段连续的1枚举长度贡献答案。

复杂度O(d(n)*n),其中d(n)为n的因子个数。

 

以上为CF官方题解,在处理01序列的时候可以采用一种较简单的写法:

枚举g,预处理c[x]表示长度i=1~x中满足g=gcd(i,n)的 i 的个数。

处理出01序列,然后将链复制一遍,倒着计算出b[i]表示从i出发最长的连续1个数。

最后枚举起点l,每个点贡献答案c[b[i]]。

#include
#include
using namespace std;const int maxn=200010;int n,a[maxn],b[maxn<<1],c[maxn],mx[maxn],gc[maxn];int gcd(int a,int b){
return !b?a:gcd(b,a%b);}//learn from LIBERATORint main(){ scanf("%d",&n); for(int i=0;i
=0;i--)if(b[i])b[i]+=b[i+1]; for(int i=0;i
View Code

 

转载于:https://www.cnblogs.com/onioncyc/p/7944055.html

你可能感兴趣的文章
android app 退出时提示确认
查看>>
win10 配置
查看>>
java 编译100个范例
查看>>
Session Cookie ServletContext
查看>>
单点登录SSO
查看>>
遇见有的软件开启后画面模糊怎么解决
查看>>
好系统重装助手教你怎么识别固态硬盘还是机械硬盘
查看>>
170. js中获取随机数 (记录一下)
查看>>
深入浅出爬虫之道: Python、Golang与GraphQuery的对比
查看>>
DHCP配置
查看>>
MySQL性能测试(二)——Ubuntu 14.4.02, MySQL 5.6.25, sysbench 4.8
查看>>
我的友情链接
查看>>
网络安全十大注意
查看>>
cisco虚拟局域网VLAN路由----待补充
查看>>
join命令实现文件内容拼接
查看>>
-bash:wget command not found的解决方法
查看>>
yara规则
查看>>
我的个人简历
查看>>
我的友情链接
查看>>
我的友情链接
查看>>