zoukankan      html  css  js  c++  java
  • 洛咕 【P1891】疯狂LCM & 三倍经验

    经验给掉先:

    经验*1

    经验*2

    经验*3

    这里给个跑得比较慢的 (n sqrt n) 预处理然后 (O(1)) 回答询问的做法

    式子

    首先我们推柿子:

    [egin{aligned}ANS&= sum_{i=1}^n lcm(i,n)\ &=sum_{i=1}^n {i* nover (i,n)} \&= nsum_{i=1}^n {iover (i,n)} \&=nsum_{d|n} sum_{i=1}^{n/d} i [(i,n/d)=1] \&=nsum_{d|n} {Big(varphi({nover d})+epsilon({nover d}) Big){nover d} over 2} \&=nsum_{d|n} {Big(varphi(d)+epsilon(d) Big){d} over 2} end{aligned} ]

    大概就是这样的

    解释一下 (varphi) 怎么出来的,我们可以看出第四行的式子如果没有 i 的话后面那玩意儿就是 $varphi $ ,然后我们考虑互质的对称性,即 当 (iperp n) 时, ((n-i)perp n) , 所以他们一一对应,并且相加为 n ,只有 (n=1) 的情况有点特别,那么咱用 单位元凑就好咯,还有除以 2 别忘了,一一对应相加后数量除以 2

    处理

    咱考虑欧拉筛一遍就能搞出这个 (varphi(i)* i) ,那么重要的就是累加了...

    咱考虑枚举每个 i ,然后再枚举一个数 j ,令 (i* j le Max_n) ,然后咱用 (ig(varphi(i)+epsilon(i)ig)i over 2) 给 $i* j $ 累加上去就好了

    这样的复杂度是多少呢?

    咱倒过来考虑,就是对于 1~n 每个数把它的因数全都计算了一遍答案,那么一个数 n 拥有的因数个数是 (O(sqrt n)) 级别的,咱积分

    [int_0^n sqrt x dx= {2over 3} n^{3over 2} =O(nsqrt n) ]

    讲道理感性理解一下就是 n 带根号啦...

    Code

    //by Judge
    #include<cmath>
    #include<cstdio>
    #include<cstring>
    #include<iostream>
    #define Rg register
    #define fp(i,a,b) for(Rg int i=(a),I=(b)+1;i<I;++i)
    #define ll long long
    using namespace std;
    const int M=1e6+3;
    typedef ll arr[M];
    #ifndef Judge
    #define getchar() (p1==p2&&(p2=(p1=buf)+fread(buf,1,1<<21,stdin),p1==p2)?EOF:*p1++)
    #endif
    char buf[1<<21],*p1=buf,*p2=buf;
    inline ll read(){ ll x=0,f=1; char c=getchar();
    	for(;!isdigit(c);c=getchar()) if(c=='-') f=-1;
    	for(;isdigit(c);c=getchar()) x=x*10+c-'0'; return x*f;
    } char sr[1<<21],z[20];int CCF=-1,Z;
    inline void Ot(){fwrite(sr,1,CCF+1,stdout),CCF=-1;}
    inline void print(ll x,char chr='
    '){
    	if(CCF>1<<20)Ot();if(x<0)sr[++CCF]=45,x=-x;
    	while(z[++Z]=x%10+48,x/=10);
    	while(sr[++CCF]=z[Z],--Z);sr[++CCF]=chr;
    } ll n,cnt; arr v,p,phi,ans;
    inline void prp(int n){ v[1]=1,phi[1]=1;
    	fp(i,2,n){ if(!v[i]) p[++cnt]=i,phi[i]=i-1;
    		for(Rg int j=1;j<=cnt&&1ll*i*p[j]<=n;++j){ v[i*p[j]]=1;
    			if(!(i%p[j])){ phi[i*p[j]]=phi[i]*p[j]; break; }
    			phi[i*p[j]]=phi[i]*(p[j]-1);
    		} phi[i]*=i,phi[i]/=2;
    	} fp(i,1,n) fp(j,1,n/i) ans[i*j]+=phi[i];
    }
    int main(){ int T=read(); prp(1e6);
    	while(T--) n=read(),print(ans[n]*n);
    	return Ot(),0;
    }
    
  • 相关阅读:
    二叉排序树(B-Tree)-c实现
    队列(Queue)-c实现
    栈(stack)--c实现(使用双链表)
    链表(list)--c实现
    c 和 指针读书笔记(1)
    c traps and pitfalls reading notes(2)
    js 控制
    正则表达式
    Android 笔记
    Android 布局方式学习
  • 原文地址:https://www.cnblogs.com/Judge/p/11268142.html
Copyright © 2011-2022 走看看