【bzoj4176】Lucas的数论 【莫比乌斯反演】【杜教筛】
题目链接
题意:求)
这是为什么呢?
考虑到)
于是我们用杜教筛算出)
=>)
=>)
=>
于是我们可以通过记忆化搜索求出前缀和。如果预先线性筛出较小的前缀和,时间复杂度大概为)。我也不会算
代码:
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
typedef long long ll;
const int N=5000005,M=1000033;
const ll mod=1000000007;
int n,p[N/10];
ll ans,tmp,mu[N];
bool vis[N];
struct HashSet{
int cnt,head[M],to[M],nxt[M];
ll v[M];
bool count(int f){
int x=f%M;
for(int i=head[x];i;i=nxt[i]){
if(to[i]==f){
return true;
}
}
return false;
}
ll get(int f){
int x=f%M;
for(int i=head[x];i;i=nxt[i]){
if(to[i]==f){
return v[i];
}
}
}
void set(int f,ll val){
int x=f%M;
to[++cnt]=f;
nxt[cnt]=head[x];
v[cnt]=val;
head[x]=cnt;
}
}s;
ll solve(int n){
if(n<=5000000){
return mu[n];
}
if(s.count(n)){
return s.get(n);
}
ll res=1;
for(int i=2,last;i<=n;i=last+1){
last=n/(n/i);
res-=solve(n/i)*(last-i+1);
res%=mod;
}
s.set(n,res);
return res;
}
ll calc(int n){
ll res=0;
for(int i=1,last;i<=n;i=last+1){
last=n/(n/i);
res+=(n/i)*(last-i+1)%mod;
res%=mod;
}
return res;
}
int main(){
mu[1]=1;
for(int i=2;i<=5000000;i++){
if(!vis[i]){
p[++p[0]]=i;
mu[i]=-1;
}
for(int j=1;j<=p[0]&&i*p[j]<=5000000;j++){
vis[i*p[j]]=true;
if(i%p[j]){
mu[i*p[j]]=-mu[i];
}else{
break;
}
}
}
for(int i=1;i<=5000000;i++){
mu[i]+=mu[i-1];
mu[i]%=mod;
}
scanf("%d",&n);
for(int i=1,last;i<=n;i=last+1){
last=n/(n/i);
tmp=calc(n/i);
ans+=tmp*tmp%mod*(solve(last)-solve(i-1))%mod;
ans%=mod;
}
printf("%lld
",(ans+mod)%mod);
return 0;
}