跳到主要内容

题解:SP12304 INVDIV - Smallest Inverse Sum of Divisors

· 阅读需 3 分钟
lailai
Blogger

原题链接

参考资料

题意简述

给定 TT 次询问,每次询问一个正整数 nn,求最小的正整数 ii 使得 σ(i)=n\sigma{(i)}=n

其中 σ(i)\sigma(i) 表示 ii 的所有正因子的和。

解题思路

说明

显然有 σ(n)n\sigma(n) \ge n,因此只需要考虑 ini \leq n 的情况即可。

无论采用哪种方法,整体思路都是:

  1. 预处理区间 [1,108][1, 10^8] 内所有整数 ii 的因子和 σ(i)\sigma(i)
  2. 在预处理的过程中,为每一个 σ(i)\sigma(i) 记录其对应的最小整数 ii
  3. 利用预处理结果快速解答所有查询。

朴素算法

分别计算每个数的 σ(i)\sigma{(i)}:枚举 i\sqrt{i} 以内的所有因子 dd,因为如果 ddii 的因子,则 id\frac{i}{d} 也是 ii 的因子。

时间复杂度 O(T+nn)O(T+n\sqrt{n}),不能通过此题。

Dirichlet 前缀和

注意到 σ(i)\sigma{(i)}id(i)\operatorname{id}(i) 的和函数,因此可以使用 Dirichlet 前缀和 进行计算。

时间复杂度 O(T+nloglogn)O(T+n\log\log{n}),实际用时 6.2s6.2\mathrm{s}

线性筛

初始化 σ(1)=1\sigma(1)=1。对于每一个素数 pp,有 σ(p)=p+1\sigma(p)=p+1

对于每一个数 nn,当通过 n×p1n'\times p_1 的方式筛掉,分类讨论:

  1. nmodp1=0n'\bmod p_1=0

根据约数和函数的性质:

σ(n)=i=1r(1+pi+pi2++piai1+piai)\sigma{(n)}=\prod_{i=1}^r(1+p_i+p_i^2+\cdots+p_i^{a_i-1}+p_i^{a_i})

进一步化简:

σ(n)=σ(n)(1+p1+p12++piai1+p1a1)(1+p1+p12++p1a11)=σ(n)f(n)f(n)\sigma{(n)}=\frac{\sigma{(n')}(1+p_1+p_1^2+\cdots+p_i^{a_i-1}+p_1^{a_1})}{(1+p_1+p_1^2+\cdots+p_1^{a_1-1})}=\frac{\sigma{(n')}f(n')}{f(n)}

同时维护一个辅助函数 f(n)f(n),表示 nn 最小质因子 p1p_1 的幂次和:

f(n)=1+p1+p12++p1a11+p1a1f(n)=1+p_1+p_1^2+\cdots+p_1^{a_1-1}+p_1^{a_1}
  1. 否则 nmodp10n'\bmod p_1\not=0

说明 nn'p1p_1 是互质的,根据积性函数性质:

σ(n)=σ(n)×σ(p1)\sigma{(n)}=\sigma{(n')}\times\sigma{(p_1)}

时间复杂度 O(T+n)O(T+n),实际用时 2.7s2.7\mathrm{s}

参考代码

Dirichlet 前缀和

#include <bits/stdc++.h>
using namespace std;

const int N=100000005;
int sigma[N],ans[N];
bool vis[N];
void init()
{
for(int i=1;i<N;i++)
{
sigma[i]=i;
}
for(int i=2;i<N;i++)
{
if(vis[i])continue;
for(int j=i;j<N;j+=i)
{
vis[j]=1;
sigma[j]+=sigma[j/i];
}
}
for(int i=1;i<N;i++)
{
if(sigma[i]<N&&!ans[sigma[i]])
{
ans[sigma[i]]=i;
}
}
}
int main()
{
ios::sync_with_stdio(false);
cin.tie(nullptr);
init();
int T;
cin>>T;
while(T--)
{
int n;
cin>>n;
cout<<(!ans[n]?-1:ans[n])<<'\n';
}
return 0;
}

线性筛

#include <bits/stdc++.h>
using namespace std;

const int N=100000005;
int prime[N],num[N],sigma[N],ans[N];
bool vis[N];
void init()
{
vis[0]=vis[1]=1;
num[1]=sigma[1]=1;
int cnt=0;
for(int i=2;i<N;i++)
{
if(!vis[i])
{
prime[++cnt]=i;
num[i]=sigma[i]=i+1;
}
for(int j=1;j<=cnt;j++)
{
if(i*prime[j]>=N)break;
vis[i*prime[j]]=1;
if(i%prime[j]==0)
{
num[i*prime[j]]=num[i]*prime[j]+1;
sigma[i*prime[j]]=sigma[i]/num[i]*num[i*prime[j]];
break;
}
num[i*prime[j]]=prime[j]+1;
sigma[i*prime[j]]=sigma[i]*sigma[prime[j]];
}
}
for(int i=1;i<N;i++)
{
if(sigma[i]<N&&!ans[sigma[i]])
{
ans[sigma[i]]=i;
}
}
}
int main()
{
ios::sync_with_stdio(false);
cin.tie(nullptr);
init();
int T;
cin>>T;
while(T--)
{
int n;
cin>>n;
cout<<(!ans[n]?-1:ans[n])<<'\n';
}
return 0;
}