【BZOJ】【3930】【CQOI2015】选数
数论/莫比乌斯反演/快速mu前缀和
比较容易想到令f[x]表示gcd=x的方案数,令g[x]表示x|gcd的方案数。
那么有$ g(d)=\sum_{d|n} f(n)$,根据莫比乌斯反演,有$f(d)=\sum_{d|n} g(n)*\mu (\frac{n}{d})$
我一开始想的是算出g以后,倒序枚举 i ,然后枚举 i 的倍数,递推出所有的f[i]……
因为g比较好算嘛……快速幂一下什么的……
然而$10^9$直接吓傻我。
Orz PoPoQQQ
快速求出mu的前缀和,$10^9$也照样不虚,太神辣
/**************************************************************
Problem: 3930
User: Tunix
Language: C++
Result: Accepted
Time:4200 ms
Memory:54024 kb
****************************************************************/ //BZOJ 3930
#include<cstdio>
#include<map>
#include<cstring>
#include<cstdlib>
#include<iostream>
#include<algorithm>
#define rep(i,n) for(int i=0;i<n;++i)
#define F(i,j,n) for(int i=j;i<=n;++i)
#define D(i,j,n) for(int i=j;i>=n;--i)
#define pb push_back
using namespace std;
typedef long long LL;
inline int getint(){
int r=,v=; char ch=getchar();
for(;!isdigit(ch);ch=getchar()) if (ch=='-') r=-;
for(; isdigit(ch);ch=getchar()) v=v*-''+ch;
return r*v;
}
const int N=1e7+,P=1e9+;
const int INF=0x3f3f3f3f;
/*******************template********************/ int mu[N],prime[],tot;
bool check[N];
map<int,LL> mu_sum;
LL n,d,l,r;
void getmu(){
int n=;
mu[]=;
F(i,,n){
if (!check[i]){
mu[i]=-;
prime[++tot]=i;
}
F(j,,tot){
int k=i*prime[j];
if (k>n) break;
check[k]=;
if (i%prime[j]) mu[k]=-mu[i];
else{
mu[k]=;
break;
}
}
}
F(i,,n) mu[i]+=mu[i-];
}
LL Mu_sum(int x){
if (x<=) return mu[x];
if (mu_sum.find(x)!=mu_sum.end())
return mu_sum[x];
LL i,last,re=;
for(i=;i<=x;i=last+){
last=x/(x/i);
if (x/i-)
re-=(Mu_sum(last)-Mu_sum(i-))*(x/i-);
}
return mu_sum[x]=re;
}
LL Pow(LL a,int b){
LL r=;
for(;b;b>>=,a=a*a%P) if (b&) r=r*a%P;
return r;
}
LL solve(){
LL i,last,re=;
for(i=;i<=r;i=last+){
last=min(r/(r/i),l/i?(l/(l/i)):INF);
re+=(Mu_sum(last)-Mu_sum(i-))*Pow(r/i-l/i,n);
re%=P;
}
return (re%P+P)%P;
}
int main(){
#ifndef ONLINE_JUDGE
freopen("3930.in","r",stdin);
freopen("3930.out","w",stdout);
#endif
n=getint(); d=getint(); l=getint(); r=getint();
l=(l-)/d; r=r/d;
getmu();
printf("%lld\n",solve());
return ;
}
然而$H-L \leq 10^5$并没用?不是的……我们可以枚举倍数!
Orz syk
记f[i]为gcd恰好为$K*i$的选数方案数,那么对于每一个$i$记$L$为$\lceil \frac{a}{K*i}\rceil $,$R$为$\lfloor\frac{b}{K*i}\rfloor$ 那么他的方案数就为$f[i] = (R-L+1) ^ N - (R-L+1)-\sum_{a=1,2,\dots} f[a*i]$最后的f[1]即为答案。注意若$\lceil \frac{a}{K} \rceil == 1$ 那么全部选K也是一种方案,需要+1。
(话说好不容易想到一次莫比乌斯反演,然而却做不出来……唉
/**************************************************************
Problem: 3930
User: Tunix
Language: C++
Result: Accepted
Time:412 ms
Memory:1664 kb
****************************************************************/ //BZOJ 3930
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<iostream>
#include<algorithm>
#define rep(i,n) for(int i=0;i<n;++i)
#define F(i,j,n) for(int i=j;i<=n;++i)
#define D(i,j,n) for(int i=j;i>=n;--i)
#define pb push_back
using namespace std;
typedef long long LL;
inline int getint(){
int r=,v=; char ch=getchar();
for(;!isdigit(ch);ch=getchar()) if (ch=='-') r=-;
for(; isdigit(ch);ch=getchar()) v=v*-''+ch;
return r*v;
}
const int N=1e5+,P=1e9+;
/*******************template********************/ int d[N],n,K,l,r,a,b;
int Pow(int a,int b){
int r=;
for(;b;b>>=,a=(LL)a*a%P) if (b&) r=(LL)r*a%P;
return r;
}
int main(){
#ifndef ONLINE_JUDGE
freopen("3930.in","r",stdin);
freopen("3930.out","w",stdout);
#endif
n=getint(); K=getint(); a=getint(); b=getint();
l=a/K; r=b/K;
if (a%K) l++;
D(i,,){
int R=r/i,L=l/i;
if (l%i) L++;
if (l<=r){
d[i]=Pow(R-L+,n);
d[i]=(d[i]-(R-L+)+P)%P;
for(int j=i+i;j<=;j+=i) d[i]=(d[i]-d[j]+P)%P;
}
}
if (l==) d[]=(d[]+)%P;
printf("%d\n",d[]);
return ;
}
3930: [CQOI2015]选数
Time Limit: 10 Sec Memory Limit: 512 MB
Submit: 434 Solved: 222
[Submit][Status][Discuss]
Description
我
们知道,从区间[L,H](L和H为整数)中选取N个整数,总共有(H-L+1)^N种方案。小z很好奇这样选出的数的最大公约数的规律,他决定对每种方
案选出的N个整数都求一次最大公约数,以便进一步研究。然而他很快发现工作量太大了,于是向你寻求帮助。你的任务很简单,小z会告诉你一个整数K,你需要
回答他最大公约数刚好为K的选取方案有多少个。由于方案数较大,你只需要输出其除以1000000007的余数即可。
Input
输入一行,包含4个空格分开的正整数,依次为N,K,L和H。
Output
输出一个整数,为所求方案数。
Sample Input
Sample Output
HINT
样例解释
Source
【BZOJ】【3930】【CQOI2015】选数的更多相关文章
- BZOJ 3930: [CQOI2015]选数 递推
3930: [CQOI2015]选数 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnline/pro ...
- 【刷题】BZOJ 3930 [CQOI2015]选数
Description 我们知道,从区间[L,H](L和H为整数)中选取N个整数,总共有(H-L+1)^N种方案.小z很好奇这样选出的数的最大公约数的规律,他决定对每种方案选出的N个整数都求一次最大公 ...
- bzoj 3930: [CQOI2015]选数
Description 我们知道,从区间[L,H](L和H为整数)中选取N个整数,总共有(H-L+1)^N种方案.小z很好奇这样选出的数的最大公约数的规律,他决定对每种方案选出的N个整数都求一次最大公 ...
- BZOJ 3930: [CQOI2015]选数 莫比乌斯反演
https://www.lydsy.com/JudgeOnline/problem.php?id=3930 https://blog.csdn.net/ws_yzy/article/details/5 ...
- 【递推】BZOJ 3930: [CQOI2015]选数
Description 我们知道,从区间[L,H](L和H为整数)中选取N个整数,总共有(H-L+1)^N种方案.小z很好奇这样选出的数的最大公约数的规律,他决定对每种方案选出的N个整数都求一次最大公 ...
- bzoj 3930: [CQOI2015]选数【快速幂+容斥】
参考:https://www.cnblogs.com/iwtwiioi/p/4986316.html 注意区间长度为1e5级别. 则假设n个数不全相同,那么他们的gcd小于最大数-最小数,证明:则gc ...
- bzoj 3930: [CQOI2015]选数【递推】
妙啊 这个题一上来就想的是莫比乌斯反演: \[ f(d)=\sum_{k=1}^{\left \lceil \frac{r}{d} \right \rceil}\mu(k)(\left \lceil ...
- BZOJ 3930: [CQOI2015]选数 莫比乌斯反演 + 杜教筛
求 $\sum_{i=L}^{R}\sum_{i'=L}^{R}....[gcd_{i=1}^{n}(i)==k]$ $\Rightarrow \sum_{i=\frac{L}{k}}^{\fra ...
- 【BZOJ】3930: [CQOI2015]选数
题意 从区间\([L, R]\)选\(N\)个数(可以重复),问这\(N\)个数的最大公约数是\(K\)的方案数.(\(1 \le N, K \le 10^9, 1 \le L \le R \le 1 ...
- 3930: [CQOI2015]选数
Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 1958 Solved: 979[Submit][Status][Discuss] Descripti ...
随机推荐
- 前端Hack之XSS攻击个人学习笔记
简单概述 ** 此篇系本人两周来学习XSS的一份个人总结,实质上应该是一份笔记,方便自己日后重新回来复习,文中涉及到的文章我都会在末尾尽可能地添加上,此次总结是我在学习过程中所写,如有任 ...
- mongodb spring 配置文件
在使用spring-data-mongodb中,需要配置spring文件,如下: mongodb.xml <?xml version="1.0" encoding=" ...
- 51nod 1584加权约数和
学到了好多东西啊这题... https://blog.csdn.net/sdfzyhx/article/details/72968468 #include<bits/stdc++.h> u ...
- zoj 3647 智商题
此题就是求格点中三角形的个数. 就是找出三点不共线的个数. n*m的矩形中有(n+1)*(m+1)个格点. 选出三个点的总个数为:C((n+1)*(m+1),3). 减掉共线的情况就是答案了. 首先是 ...
- CentOS下KVM增加磁盘/磁盘扩容/在线扩容
一.磁盘镜像操作(适用于raw和qcow2格式) 1.创建镜像 qemu-img create -f qcow2(格式) /kvm/centos1_1.qcow2(路径) 5G(容量) 2.修改镜像容 ...
- 大型电商业务架构 IT大咖说 - 大咖干货,不再错过
大型电商业务架构 IT大咖说 - 大咖干货,不再错过 http://www.itdks.com/dakashuo/new/dakalive/detail/591
- memcached对key和value的限制 memcached的key最大长度和Value最大长度
memcached的简单限制就是键(key)和item的限制.最大键长为250个字符.可以接受的储存数据不能超过1MB,因为这是典型slab 的最大值.这里我们可以突破对key长度的限制.问题解决:修 ...
- SIMATIC PID温度控制
SIMATIC PID温度控制 // VAR_INPUT ------------------------------------------------------------------- #if ...
- LDO current regulator for power LED
LDO current regulator for power LED Challenge You've got a power LED? Great! Build a flash light! Wh ...
- Git 修复 bug 切换分支时,如何保存修改过的代码(即如何保存现场)?
工作除了开发最新的版本之外还要对原来的版本做例行的维护,修修补补.于是有了在两个分支之间游走切换的问题,最新改版的代码在分支 new 上,旧版本的代码在分支 old 上,我在 new 上开发了一半,忽 ...