[SDOI2015]约数个数和

题面

∑ i = 1 n ∑ j = 1 m d ( i j ) \sum_{i=1}^n\sum_{j=1}^md(ij) i=1nj=1md(ij)
d ( x ) d(x) d(x)表示x的约数个数

题解

其实写这篇题解主要是为了证明下面这个式子:
d ( i j ) = ∑ x ∣ i ∑ y ∣ j [ g c d ( i , j ) = 1 ] d(ij)=\sum_{x|i}\sum_{y|j}[gcd(i,j)=1] d(ij)=xiyj[gcd(i,j)=1]
证明完这个式子后直接带入就很容易莫比乌斯反演了,之后的过程比较套路,就不写了,直接放证明。

证明:

i = 2 a 1 3 a 2 5 a 3 . . . p i a i . . . i=2^{a_1}3^{a_2}5^{a_3}...p_i^{a_i}... i=2a13a25a3...piai...
j = 2 b 1 3 b 2 5 b 3 . . . p i b i . . . j=2^{b_1}3^{b_2}5^{b_3}...p_i^{b_i}... j=2b13b25b3...pibi...

那么对于每个ij的约数x, x = 2 c 1 3 c 2 5 c 3 . . . p i c i . . . x=2^{c_1}3^{c_2}5^{c_3}...p_i^{c_i}... x=2c13c25c3...pici...

a i + b i ≥ c i ≥ 0 a_i+b_i\ge c_i\ge 0 ai+bici0

也就是 c i ∈ [ 0... a i + b i ] c_i\in [0...a_i+b_i] ci[0...ai+bi]

如果 c i ∈ [ 0... a i ] c_i\in [0...a_i] ci[0...ai],那么 c i ≤ a i c_i\le a_i ciai p i c i ∣ p i a i p_i^{c_i}|p_i^{a_i} picipiai
如果 c i ∈ ( a i . . . a i + b i ] c_i\in (a_i...a_i+b_i] ci(ai...ai+bi],那么 c i − a i ≤ b i c_i-a_i\le b_i ciaibi p i c i − a i ∣ p i b i p_i^{c_i-a_i}|p_i^{b_i} piciaipibi

也就是说对于每个 p i p_i pi,x一定可以要么映射成i的因子,要么映射成j的因子。

那么将每个质因子如此映射,所以每个ij的因子都可以分成两部分,一部分完全是i的因子,一部分完全是j的因子,这两部分一定互质(因为对于任意i, a i a_i ai b i b_i bi只有一个大于零)

证毕。

放代码:

#include
using namespace std; 
template<typename tn> void read(tn &a){
     
	tn x=0,f=1;char c=' ';
	for(;!isdigit(c);c=getchar()) if(c=='-') f=-1;
	for(; isdigit(c);c=getchar()) x=x*10+c-'0';
	a=x*f;
}
const int N=5e4+5;
#define int long long
bool is_prime[N];
int prime[N>>2],cnt,mu[N],Sum[N],dil[N];
int T,n,m;
signed main(){
     
	mu[1]=1;
	for(int i=2;i<N;i++){
     
		if(!is_prime[i]) prime[++cnt]=i,mu[i]=-1;
		for(int j=1;j<=cnt&&prime[j]*i<N;j++){
     
			is_prime[prime[j]*i]=1;
			if(i%prime[j]==0){
     
				mu[i*prime[j]]=0;
				break;
			}
			mu[prime[j]*i]=-mu[i];
		}
	}
	for(int i=1;i<N;i++) Sum[i]=Sum[i-1]+mu[i];
	for(int i=1;i<N;i++)
		for(int l=1,r;l<=i;l=r+1){
     
			r=i/(i/l);
			dil[i]+=(r-l+1)*(i/l); 
		}
	read(T);
	while(T--){
     
		read(n);
		read(m);
		int k=min(n,m);
		int ans=0;
		for(int l=1,r;l<=k;l=r+1){
     
			r=min(n/(n/l),m/(m/l));
			ans+=(Sum[r]-Sum[l-1])*dil[n/l]*dil[m/l];
		}
		printf("%lld\n",ans);
	}
	return 0;
}

你可能感兴趣的:(题解,数论数学)