UVA 三道 GCD 题 题解

本文探讨了求解∑i=1n∑j=i+1ngcd(i,j)的三种方法,分别适用于小规模、中等规模和大规模数据集。介绍了O(n^2)暴力解法、枚举GCD并利用欧拉函数优化至O(n),以及针对更大数据集采用莫比乌斯反演和整除分块的高效算法。

问题都是求 ∑i=1n∑j=i+1ngcd⁡(i,j)\sum\limits_{i=1}^n\sum\limits_{j=i+1}^n \gcd(i,j)i=1nj=i+1ngcd(i,j)

总结一下套路

首先第一题直接 O(n2)O(n^2)O(n2) 暴力就能解决

int n, ans ;

signed main(){
	while (scanf("%d", &n) != EOF && n) {
		ans = 0 ;
		rep(i, 1, n) rep(j, i + 1, n) ans += __gcd(i, j) ;
		printf("%d\n", ans) ;
	}
	return 0 ;
}

对于第二题, n&lt;=200000n &lt;= 200000n<=200000

我们枚举 gcd⁡\gcdgcdkkk,现在的问题就是有多少对, 我们记做 f(k)f(k)f(k), 那么答案就是 ∑i=1n(f(i)−1)∗i\sum\limits_{i=1}^n(f(i)-1)*ii=1n(f(i)1)i,注意要 −1-11 因为不存在 gcd⁡(i,i)=i\gcd(i,i)=igcd(i,i)=i 的情况

f(k)f(k)f(k) 怎么求?

若 a,b 互质, 则 gcd⁡(ak,bk)=k\gcd(ak,bk)=kgcd(ak,bk)=k

枚举 a,ba,ba,b 中较大的一个,记做 iii, 那么另一个数有 φ(i)\varphi(i)φ(i)种可能,所以 f(k)=∑i=1n/kφ(i)f(k)=\sum\limits_{i=1}^{n/k}\varphi(i)f(k)=i=1n/kφ(i),用前缀和处理

复杂度 O(n)O(n)O(n)

int n, cnt ;
ll phi[N], f[N], s[N], prime[N / 10] ;
ll ans ;

void sieve(int n) {
	phi[1] = 1 ;
	rep(i, 2, n) {
        if (!f[i]) {
        	prime[++cnt] = i ;
        	phi[i] = i - 1 ;
		}
		for (int j = 1; j <= cnt && prime[j] * i <= n; j++) {
			f[i * prime[j]] = 1 ;
			if (i % prime[j]) phi[i * prime[j]] = phi[i] * (prime[j] - 1) ;
			else {
				phi[i * prime[j]] = phi[i] * prime[j] ;
				break ;
			}
		}
	}
}

void init(int n) {
	clr(f) ;
	rep(i, 1, n)
	for (int j = 2 * i; j <= n; j += i)
	f[j] += i * phi[j / i] ;
	rep(i, 1, n) s[i] = s[i - 1] + f[i] ;
}

signed main() {
	sieve(N - 10) ;
	init(N - 10) ;
	while (scanf("%d", &n) != EOF && n) printf("%lld\n", s[n]) ;
}

对于第三题, n&lt;=4000000n&lt;=4000000n<=4000000

其实本质就是 ZAPZAPZAP 那题一样的,直接莫比乌斯反演+整除分块就行了

时间复杂度 O(n)O(\sqrt n)O(n)

评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值