欧拉函数模板
关于欧拉函数,我们首先介绍他到两条性质,这两条性质对于下面到推到至关重要
(a)如果p是素数且k
≥
≥
1,则
ϕ(pk)=pk−pk−1
ϕ
(
p
k
)
=
p
k
−
p
k
−
1
(b)如果
gcd(m,n)=1
g
c
d
(
m
,
n
)
=
1
,则
ϕ(mn)=ϕ(m)ϕ(n)
ϕ
(
m
n
)
=
ϕ
(
m
)
ϕ
(
n
)
下面推到欧拉函数公式
根据算数基本定理,每个数字都可分解成素数到乘积形式
设
x=pk11pk22...pkrr
x
=
p
1
k
1
p
2
k
2
.
.
.
p
r
k
r
ϕ(x)=ϕ(pk11)ϕ(pk22)...ϕ(pkrr)
ϕ
(
x
)
=
ϕ
(
p
1
k
1
)
ϕ
(
p
2
k
2
)
.
.
.
ϕ
(
p
r
k
r
)
——性质(a)
=(pk11−pk1−11)⋅(pk22−pk2−12)...(pkrr−pkr−1r)
=
(
p
1
k
1
−
p
1
k
1
−
1
)
⋅
(
p
2
k
2
−
p
2
k
2
−
1
)
.
.
.
(
p
r
k
r
−
p
r
k
r
−
1
)
——–性质(b)
=pk11pk22...pkrr⋅(1−1p1)...(1−1pr)
=
p
1
k
1
p
2
k
2
.
.
.
p
r
k
r
⋅
(
1
−
1
p
1
)
.
.
.
(
1
−
1
p
r
)
=x⋅(p1−1p1)⋅(p2−1p2)...(pr−1pr)
=
x
⋅
(
p
1
−
1
p
1
)
⋅
(
p
2
−
1
p
2
)
.
.
.
(
p
r
−
1
p
r
)
得到了公式,代码怎么写?
其实我们到代码是基于素数筛选,写法很类似
相信大家看了上面的推导代码会好理解些
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;
}