欧拉函数模板
关于欧拉函数,我们首先介绍他到两条性质,这两条性质对于下面到推到至关重要
(a)如果p是素数且k≥≥1,则 ϕ(pk)=pk−pk−1ϕ(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)
=(pk11−pk1−11)⋅(pk22−pk2−12)...(pkrr−pkr−1r)=(p1k1−p1k1−1)⋅(p2k2−p2k2−1)...(prkr−prkr−1) ——–性质(b)
=pk11pk22...pkrr⋅(1−1p1)...(1−1pr)=p1k1p2k2...prkr⋅(1−1p1)...(1−1pr)
=x⋅(p1−1p1)⋅(p2−1p2)...(pr−1pr)=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;
}
本文介绍了欧拉函数的基本性质及推导过程,并给出了两种不同的实现方法:一种是基于素数筛选的欧拉函数表构建,另一种是针对单个数值的直接计算。
4万+

被折叠的 条评论
为什么被折叠?



