UVA 三道 GCD 题 题解

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

总结一下套路

首先第一题直接 O ( n 2 ) 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; = 200000 n &lt;= 200000 n<=200000

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

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

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

枚举 a , b a,b a,b 中较大的一个,记做 i i i, 那么另一个数有 φ ( i ) \varphi(i) φ(i)种可能,所以 f ( k ) = ∑ i = 1 n / 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; = 4000000 n&lt;=4000000 n<=4000000

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

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

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值