【转】UVALive 5964 LCM Extreme --欧拉函数
题目大意:求lcm(1,2)+lcm(1,3)+lcm(2,3)+....+lcm(1,n)+....+lcm(n-2,n)+lcm(n-1,n)
解法:设sum(n)为sum(lcm(i,j))(1<=i<j<=n)之间最小公倍数的和,
f(n)为sum(i*n/gcd(i,n))(1<=i<n)
那么sum(n)=sum(n-1)+f(n)。
可以用线性欧拉筛选+递推来做。
代码:
#include <iostream> #include <cstdio> #include <cstring> #include <cmath> #include <algorithm> using namespace std; #define N 207 typedef unsigned long long LL; const int maxn=5000005; LL phi[maxn],sum[maxn],f[maxn]; void Euler() { memset(phi,0,sizeof(phi)); int i,j; phi[1]=1; for(i=2;i<maxn;i++) { if(phi[i]) continue; for(j=i;j<maxn;j+=i) { if(!phi[j]) phi[j]=j; phi[j]=phi[j]/i*(i-1); } } for(i=1;i<maxn;i++) phi[i]=phi[i]*i/2; //与i互质的数之和 } void init() { Euler(); memset(sum,0,sizeof(sum)); memset(f,0,sizeof(f)); int i,j; sum[1]=f[1]=0; for(i=2;i<maxn;i++) { f[i]+=phi[i]*i; //与i互质的数之间的lcm之和 for(j=2*i;j<maxn;j+=i) f[j]+=phi[i]*j; //gcd(x,j)=i的sum(lcm(x,j)) sum[i]=sum[i-1]+f[i]; } } int main() { init(); int t,icase=0,n; scanf("%d",&t); while(t--) { scanf("%d",&n); printf("Case %d: %llu\n",++icase,sum[n]); } return 0; }
作者:whatbeg
出处1:http://whatbeg.com/
出处2:http://www.cnblogs.com/whatbeg/
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。
更多精彩文章抢先看?详见我的独立博客: whatbeg.com