【BZOJ】【P4407】【于神之怒加强版】【题解】【数论】
2016-02-16 08:47
302 查看
传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=4407
这两天刚好在给学弟讲数论,bzoj上就来了一道裸题……
答案就是\sum_D F(D)*n/d*m/d
F(D=\sum{d|D} d^K mu(D/d)
线性筛,分块求
F(p)=p^K-1
F(x*p)=F(x)*F(p) (x,p)=1
F(x*p)=F(x)*p (x,p)/=1
Code:
#include<bits/stdc++.h>
using namespace std;
const int maxn=5e6+10;
typedef long long LL;
const LL MOD=1e9+7;
int T,K;
bool notp[maxn];
LL p[maxn];
LL f[maxn];
LL sum[maxn];
LL pw(LL x,LL k,LL p){
LL ans=1;
for(;k;k>>=1){
if(k&1)ans=ans*x%p;
x=x*x%p;
}return ans;
}
void pre(){
f[1]=1;
for(int i=2;i<maxn;i++){
if(!notp[i]){
f[i]=pw(i,K,MOD)-1;
p[++p[0]]=i;
}
for(int j=1;j<=p[0];j++){
if(i*p[j]>=maxn)break;
notp[i*p[j]]=1;
if(i%p[j]==0){
f[i*p[j]]=f[i]*pw(p[j],K,MOD)%MOD;
break;
}else{
f[i*p[j]]=f[i]*f[p[j]]%MOD;
}
}
}
for(int i=1;i<maxn;i++)
sum[i]=(sum[i-1]+f[i])%MOD;
}
int solve(int n,int m){
LL ans=0;
for(int i=1,j=0;i<=n;i=j+1){
j=min(n/(n/i),m/(m/i));
ans+=(sum[j]-sum[i-1]+MOD)%MOD*(n/i)%MOD*(m/i)%MOD;
ans%=MOD;
}return ans%MOD;
}
int main(){
scanf("%d%d",&T,&K);
pre();
// for(int i=1;i<=20;i++)
// cout<<i<<" "<<f[i]<<endl;
while(T--){
int n,m;scanf("%d%d",&n,&m);
if(n>m)swap(n,m);
printf("%d\n",int(solve(n,m)%MOD));
}
return 0;
}
这两天刚好在给学弟讲数论,bzoj上就来了一道裸题……
答案就是\sum_D F(D)*n/d*m/d
F(D=\sum{d|D} d^K mu(D/d)
线性筛,分块求
F(p)=p^K-1
F(x*p)=F(x)*F(p) (x,p)=1
F(x*p)=F(x)*p (x,p)/=1
Code:
#include<bits/stdc++.h>
using namespace std;
const int maxn=5e6+10;
typedef long long LL;
const LL MOD=1e9+7;
int T,K;
bool notp[maxn];
LL p[maxn];
LL f[maxn];
LL sum[maxn];
LL pw(LL x,LL k,LL p){
LL ans=1;
for(;k;k>>=1){
if(k&1)ans=ans*x%p;
x=x*x%p;
}return ans;
}
void pre(){
f[1]=1;
for(int i=2;i<maxn;i++){
if(!notp[i]){
f[i]=pw(i,K,MOD)-1;
p[++p[0]]=i;
}
for(int j=1;j<=p[0];j++){
if(i*p[j]>=maxn)break;
notp[i*p[j]]=1;
if(i%p[j]==0){
f[i*p[j]]=f[i]*pw(p[j],K,MOD)%MOD;
break;
}else{
f[i*p[j]]=f[i]*f[p[j]]%MOD;
}
}
}
for(int i=1;i<maxn;i++)
sum[i]=(sum[i-1]+f[i])%MOD;
}
int solve(int n,int m){
LL ans=0;
for(int i=1,j=0;i<=n;i=j+1){
j=min(n/(n/i),m/(m/i));
ans+=(sum[j]-sum[i-1]+MOD)%MOD*(n/i)%MOD*(m/i)%MOD;
ans%=MOD;
}return ans%MOD;
}
int main(){
scanf("%d%d",&T,&K);
pre();
// for(int i=1;i<=20;i++)
// cout<<i<<" "<<f[i]<<endl;
while(T--){
int n,m;scanf("%d%d",&n,&m);
if(n>m)swap(n,m);
printf("%d\n",int(solve(n,m)%MOD));
}
return 0;
}
相关文章推荐
- BZOJ3275 Number (最小割)
- [bzoj1003] [ZJOI2006]物流运输trans
- [bzoj1500][NOI2005]维修数列
- [bzoj1208] [HNOI2004]宠物收养所
- [bzoj1269][AHOI2006]文本编辑器editort
- [bzoj1503][NOI2004]郁闷的出纳员
- bzoj4305 数学
- bzoj3926 广义后缀自动机
- bzoj2780 广义后缀自动机+parent树+Dfs序+树状数组
- BZOJ1997 2-sat
- bzoj4027 贪心
- [BZOJ2038][2009国家集训队][莫队][分块]小z的袜子
- [BZOJ2594][WC2006][LCT][MST]水管局长数据加强版
- [BZOJ2300][HAOI2011][动态凸包]防线修建
- [BZOJ1045][HAOI2008][贪心]糖果传递
- [BZOJ2539][CTSC2000][KM]丘比特的烦恼
- [BZOJ1004][HNOI2008][Burnside引理][DP]Cards
- [BZOJ1202][HNOI2005][并查集]狡猾的商人
- [BZOJ1179][APIO2009][Tarjan][拓扑排序][递推]Atm
- [BZOJ1095][ZJOI2007][线段树]Hide捉迷藏