• BZOJ 4804: 欧拉心算 欧拉函数


    Description

     给出一个数字N

    Input

    第一行为一个正整数T,表示数据组数。
    接下来T行为询问,每行包含一个正整数N。
    T<=5000,N<=10^7
    题解: 求 $sum_{i=1}^{n}sum_{j=1}^{n}varphi(gcd(i,j))$
     
    $=sum_{d=1}^{n}varphi(d)sum_{i=1}^{n}sum_{j=1}^{n}[gcd(i,j)==d]$
     
    $=sum_{d=1}^{n}varphi(d)sum_{i=1}^{frac{n}{d}}sum_{j=1}^{frac{n}{d}}[gcd(i,j)==1]$
     
    其中 $sum_{i=1}^{n}sum_{j=1}^{n}[gcd(i,j)==1]=2 imessum_{i=1}^{n}varphi(i)-1$
     
    可以用数形结合来思考:
     
    $sum_{i=1}^{n}sum_{j=1}^{i}[gcd(i,j)==1]=sum_{i=1}^{n}varphi(i)$
     
    这个式子可以看作是对角线一半的欧拉函数值(以 $i$ 为横坐标,$j$ 为纵坐标)
     
    而 $sum_{i=1}^{n}sum_{j=1}^{n}[gcd(i,j)==1]$ 正好是对角线折叠过去的结果.
     
    不过还要减去 $(1,1)$ 多出来的贡献(只有 $1$ 是和自己本身互质的数)
     
    令 $Sum(n)=sum_{i=1}^{n}varphi(i)$
     
    答案为 $sum_{d=1}^{n}varphi(d)sum_{i=1}^{frac{n}{d}}sum_{j=1}^{frac{n}{d}}[gcd(i,j)==1]$
     
    $Rightarrowsum_{d=1}^{n}varphi(d) imes(2 imessum_{i=1}^{frac{n}{d}}varphi(i)-1)$
     
    $Rightarrowsum_{d=1}^{n}varphi(d) imes(2 imes Sum(frac{n}{d})-1)$
     
    $Rightarrow2 imessum_{d=1}^{n}varphi(d)Sum(frac{n}{d})-Sum(n)$
     
    提前与处理一下欧拉函数前缀和即可.
    #include<bits/stdc++.h>
    #define maxn 10000003 
    #define M 10000000
    #define ll long long 
    using namespace std;
    void setIO(string s)
    {
    	string in=s+".in"; 
    	freopen(in.c_str(),"r",stdin); 
    }
    int cnt; 
    bool vis[maxn];
    int prime[maxn], phi[maxn];  
    ll sumv[maxn];  
    inline void Linear_shaker()
    {
    	phi[1]=1; 
    	int i,j; 
    	for(i=2;i<=M;++i)
    	{
    		if(!vis[i]) prime[++cnt]=i, phi[i]=i-1; 
    		for(j=1;j<=cnt&&1ll*i*prime[j]<=M;++j)
    		{
    			vis[i*prime[j]]=1; 
    			if(i%prime[j]==0) 
    			{
    				phi[i*prime[j]]=phi[i]*prime[j]; 
    				break; 
    			}
    			phi[i*prime[j]]=phi[i]*(prime[j]-1); 
    		}
    	} 
    	for(i=1;i<=M;++i) sumv[i]=sumv[i-1]+phi[i]; 
    }
    inline void solve()
    {
    	int n,i,j; 
    	scanf("%d",&n); 
    	ll re=0; 
    	for(i=1;i<=n;i=j+1)
    	{
    		j=(n/(n/i)); 
    		re+=(sumv[j]-sumv[i-1])*(sumv[n/i]*2-1); 
    	}
    	printf("%lld
    ",re); 
    }
    int main()
    {
    	// setIO("input"); 
    	Linear_shaker(); 
    	int T; 
    	scanf("%d",&T);
    	while(T--) solve();
    	return 0; 
    }
    

      

  • 相关阅读:
    springBoot 2.1.5 pom 文件 unknown 错误
    @HystrixCommand 不能被导包
    SQL数据库连接语句
    ADO.NET中COMMAND对象的ExecuteNonQuery、ExcuteReader和ExecuteScalar方法
    重载和重写的区别
    抽象类和接口的相同点和不同点
    结构详解
    简单工厂和抽象工厂的区别
    DataRead和DataSet的异同
    什么是Web Server
  • 原文地址:https://www.cnblogs.com/guangheli/p/11097308.html
Copyright © 2020-2023  润新知