The Euler function HDU杭电2824 【欧拉函数打表】

Problem Description
The Euler function phi is an important kind of function in number theory, (n) represents the amount of the numbers which are smaller than n and coprime to n, and this function has a lot of beautiful characteristics. Here comes a very easy question: suppose you are given a, b, try to calculate (a)+ (a+1)+....+ (b)
 

Input
There are several test cases. Each line has two integers a, b (2<a<b<3000000).
 

Output
Output the result of (a)+ (a+1)+....+ (b)
 

Sample Input
  
  
3 100
 

Sample Output
  
  
3042
 

题意:x属于1到n,把f(x)都加起来(f(x)是从一到n有几个数和n互质)


/*
若N是质数p的k次幂(即N=p^k),φ(n)=p^k-p^(k-1)=(p-1)p^(k-1)。
若m,n互质,φ(mn)=(m-1)(n-1)=φ(m)φ(n)

利用欧拉函数如下性质,可以快速求出欧拉函数的值(a为N的质因素)

  若( N%a ==0&&(N/a)%a ==0)则有:E(N)= E(N/a)*a;

  若( N%a ==0&&(N/a)%a !=0)则有:E(N)= E(N/a)*(a-1);

*/

#include <stdio.h>
#include <string.h> 
#define N 3000300
int prime[N],isprime[N];
int ans[N];
void Is_Prime()//筛素数 
{
	for(int i=2;i<N;++i)
	{
		for(int j=i*2;j<N;j+=i)
		{
			isprime[j]=0;
		}
	}
}
void get_ans(){
    int i,j,cnt=0;
    for(i=2;i<N;i++)
	{
        if(isprime[i])
		{
            prime[cnt++]=i;//存素数 
            ans[i]=i-1;
        }
        for(j=0;j<cnt && i*prime[j]<N;j++)//注意这里,i*prime[j]<N 可换成 prime[j]<=N/i(带等号)
		{     
            if(i%prime[j]==0)
                ans[i*prime[j]]=ans[i]*prime[j];
            else
                ans[i*prime[j]]=ans[i]*(prime[j]-1);
        }
    }
}
int main()
{
	int a,b;
	memset(isprime,1,sizeof(isprime)); 
	Is_Prime();
	get_ans();
	while(~scanf("%d%d",&a,&b))
	{
		__int64 sum=0;//要用64位的或long long型 
		for(int i=a; i<=b; ++i)
		{
			sum+=ans[i];
		}
		printf("%I64d\n",sum);
	}
	return 0;
}


方法二: 这个方法应该更好懂


#include <stdio.h>
#include <string.h>
#define size 3000300
int euler[size];
void Init()
{ 
	memset(euler,0,sizeof(euler)); 
	euler[1]=1;
	for(int i=2;i<size;i++)
		if(!euler[i])//euler[i]为零时要么没处理,要么i就是素数 
			for(int j=i;j<size;j+=i)
			{
				if(!euler[j])//当euler[j]为零的时候,j的质因子就只有i了 
				{
					euler[j]=j/i*(i-1);; 
				} 
				else euler[j]=euler[j]/i*(i-1);//先进行除法是为了防止中间数据的溢出 
			} 
}
int main()
{
	int T;
	int N,M;
	Init();
	//scanf("%d",&T);
	while(~scanf("%d%d",&N,&M))
	{
		;
		__int64 sum=0;
		for(int i=N;i<=M;++i)
			sum+=euler[i];
		printf("%I64d\n",sum);
	}
	return 0;
}


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值