zoukankan      html  css  js  c++  java
  • P1891 疯狂LCM

    题意

    (sum_{i=1}^noperatorname{lcm}(i,n))
    多测,(Tleq 3cdot 10^5,1leq nleq 10^6)


    题解

    看这数据范围也能知道是个预处理+(O(1))查询之类的东西
    很容易想到的变形:

    [sum_{i=1}^noperatorname{lcm}(i,n)=sum_{i=1}^nfrac{icdot n}{gcd(i,n)}=nsum_{i=1}^nfrac{i}{gcd(i,n)} ]

    然后就不会了

    其实是要加一层(sum),这个不看题解是真想不到,不过这可能是个挺套路的东西?

    [nsum_{d | n}sum_{i=1}^nfrac{i}{d}cdot[gcd(i,n)=d] ]

    虽然看起来比较废话但很关键

    然后再变形,中括号里除以(d)变成(gcd(dfrac{i}{d},dfrac{n}{d})=1),其实就是如果(dfrac{i}{d})(dfrac{n}{d})互质,那么他们都乘(d)以后的(gcd)就是(d)

    [nsum_{d | n}sum_{i=1}^nfrac{i}{d}cdot[gcd(frac{i}{d},frac{n}{d})=1] ]

    然后让(i)替代(dfrac{i}{d})

    [nsum_{d|n}sum_{i=1}^{frac{n}{d}}icdot[gcd(i,frac{n}{d})=1] ]

    又因为这个(dfrac{n}{d},d)(n)的“一对”因数,所以这个式子可以直接写成一个更好看的形式:

    [nsum_{d|n}sum_{i=1}^{d}icdot[gcd(i,d)=1] ]

    然后这就有点欧拉(varphi)函数的意思了
    关键是那个(i)很碍眼,所以当然是要再看一波题解

    由于(gcd(i,d)=gcd(d-i,d))
    这个和欧几里得法求(gcd)很像,具体证明可以去这里看
    所以考虑(sum_{i=1}^{d}icdot[gcd(i,d)=1])里的每一个(i)(d-i),如果(gcd(i,d) eq 1)肯定就不用管,如果等于(1),那么这两项的值就是(d)
    那么有多少(gcd(i,d)=gcd(d-i,d)=1)?肯定是(dfrac{varphi(d)}{2}),注意这里说的是多少
    但是很容易发现,(d=1)是上面式子成(0)了,不成立,所以应该给(sum)中的分子加上(1)然后下取整,就避免了这个情况
    然后更准确的答案就表述为:

    [nsum_{d|n}lfloorfrac{varphi(d)cdot d+1}{2} floor ]

    其实这时已经可以(O(sqrt n))回答了,但还可以做的更好

    (O(nlog log n))跑一个类似于埃氏筛的过程,我们设(f_j=sum_{d|j}lfloordfrac{varphi(d)cdot d+1}{2} floor),应该都知道埃氏筛里面(j)一般代表啥吧
    然后对于每一个枚举到的(i),实际上就是上式中的每一个(d),然后按公式给(f_j)加上就好


    小总结

    当然我做的题太少可能说的不全
    其实这种(gcd,operatorname{lcm})有关的问题,很多都是转变成([gcd=cdots] imes cdots)的形式,(cdots)都是数,然后和(varphi)结合起来
    有时这个转换的方式就是通过加上一个(sum_{d|n}),其它的还没遇见,遇见了可能会写在这里
    反正(gcd)这种最容易联想到的就是(varphi)

    #include<cstdio>
    #include<algorithm>
    #include<iostream>
    #include<cmath>
    #include<iomanip>
    #include<cstring>
    #define reg register
    #define EN std::puts("")
    #define LL long long
    inline int read(){
    	register int x=0;register int y=1;
    	register char c=std::getchar();
    	while(c<'0'||c>'9'){if(c=='-') y=0;c=std::getchar();}
    	while(c>='0'&&c<='9'){x=x*10+(c^48);c=std::getchar();}
    	return y?x:-x;
    }
    int n=1e6;
    LL phi[1000006],f[1000006];
    int prime[500006],notprime[1000006];
    inline void get_phi(){
    	phi[1]=1;
    	for(reg int i=2;i<=n;i++){
    		if(!notprime[i]) prime[++prime[0]]=i,phi[i]=i-1;
    		for(reg int j=1;j<=prime[0]&&i*prime[j]<=n;j++){
    			notprime[i*prime[j]]=1;
    			if(!(i%prime[j])){
    				phi[i*prime[j]]=prime[j]*phi[i];
    				break;
    			}
    			else phi[i*prime[j]]=phi[i]*(prime[j]-1);
    		}
    	}
    }
    int main(){
    	get_phi();
    	for(reg int i=1;i<=n;i++)
    		for(reg int j=i;j<=n;j+=i)
    			f[j]+=(phi[i]*i+1)>>1;//要加1,如果i=1的话式子就是1了符合要求,如果不是和没加一样不用管 
    	int T=read();while(T--){
    		n=read();
    		std::printf("%lld
    ",f[n]*n);
    	}
    	return 0;
    }
    
  • 相关阅读:
    echarts 柱状图移除圆角
    echarts 图例显示到右边
    css图片文字一排
    linux磁盘分区、挂载、查看
    mysql的sql筛选排重最大值并修改其属性
    Java调用Linux下的shell命令并将结果以流的形式返回
    SHELL脚本中执行SQL语句操作MYSQL的5种方法
    在 Mac 上使用 `sed` 命令的几个坑
    Linux sed命令实现替换文本内容
    docker ps -a
  • 原文地址:https://www.cnblogs.com/suxxsfe/p/12624033.html
Copyright © 2011-2022 走看看