欧拉函数模板

本文介绍了欧拉函数的基本性质及推导过程,并给出了两种不同的实现方法:一种是基于素数筛选的欧拉函数表构建,另一种是针对单个数值的直接计算。

欧拉函数模板

关于欧拉函数,我们首先介绍他到两条性质,这两条性质对于下面到推到至关重要
(a)如果p是素数且k1,则 ϕ(pk)=pkpk1ϕ(pk)=pk−pk−1
(b)如果gcd(m,n)=1gcd(m,n)=1,则ϕ(mn)=ϕ(m)ϕ(n)ϕ(mn)=ϕ(m)ϕ(n)

下面推到欧拉函数公式
根据算数基本定理,每个数字都可分解成素数到乘积形式
x=pk11pk22...pkrrx=p1k1p2k2...prkr
ϕ(x)=ϕ(pk11)ϕ(pk22)...ϕ(pkrr)ϕ(x)=ϕ(p1k1)ϕ(p2k2)...ϕ(prkr) ——性质(a)
=(pk11pk111)(pk22pk212)...(pkrrpkr1r)=(p1k1−p1k1−1)⋅(p2k2−p2k2−1)...(prkr−prkr−1) ——–性质(b)
=pk11pk22...pkrr(11p1)...(11pr)=p1k1p2k2...prkr⋅(1−1p1)...(1−1pr)
=x(p11p1)(p21p2)...(pr1pr)=x⋅(p1−1p1)⋅(p2−1p2)...(pr−1pr)

得到了公式,代码怎么写?
其实我们到代码是基于素数筛选,写法很类似
相信大家看了上面的推导代码会好理解些
code:

//筛选法打欧拉函数表     
#define Max 1000001    
int euler[Max];    
void Init(){     
     euler[1]=1;    
     for(int i=2;i<Max;i++)    
       euler[i]=i;    
     for(int i=2;i<Max;i++)    
        if(euler[i]==i)    
           for(int j=i;j<Max;j+=i)    
              euler[j]=euler[j]/i*(i-1);//先进行除法是为了防止中间数据的溢出     
}    

还有一种直接求的方式

//直接求解欧拉函数    
int euler(int n){ //返回euler(n)     
     int res=n,a=n;    
     for(int i=2;i*i<=a;i++){    
         if(a%i==0){    
             res=res/i*(i-1);//先进行除法是为了防止中间数据的溢出     
             while(a%i==0) a/=i;    
         }    
     }    
     if(a>1) res=res/a*(a-1);    
     return res;    
}    
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值