题目链接
题意:求模的值。为n的约数个数。
题解:我们有一个结论:
这是为什么呢?
考虑到的任何一个约数都可以表示成满足,的的形式,上面的式子就是枚举i,j的值的过程。那为什么要求呢?如果有d满足那么,重复计算了。所以我们有了上面的式子。
于是我们用杜教筛算出的前缀和,再分块计算即可。
杜教筛
,因为只有i=1时有贡献1。
=>
=>
=>,即把带入。
于是我们可以通过记忆化搜索求出前缀和。如果预先线性筛出较小的前缀和,时间复杂度大概为。我也不会算
代码:
#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;
}