数论/the first wave

时间:2023-03-09 15:57:16
数论/the first wave

线性筛素数(原来我之前学的不是线性的啊。。。

void getprime(){
rep(i,2,nmax){
if(!vis[i]) prime[++prime[0]]=i;
for(int j=1;j<=prime[0]&&i*prime[j]<=nmax;j++){
vis[i*prime[j]]=true;
if(i%prime[j]==0) break;
}
}
}

利用了每个合数必有一个最小素因子,每个合数仅被它的最小素因子筛去正好一次,所以是线性时间。
代码中体现在: if(i%prime[j]==0) break;(抄自M.J的blog

求欧拉函数

int getphi(int x){
int ans=x;
for(int i=2;i*i<=x;i++){
if(x%i==0) ans=ans/i*(i-1);
while(x%i==0) x/=i;
}
if(x!=1) ans=ans/x*(x-1);
return ans;
}

线性筛素数,欧拉函数。

欧拉函数性质:f(ab)=f(a)f(b);

void getphi(){
phi[1]=1;
rep(i,2,nmax){
if(!vis[i]) prime[++prime[0]]=i,phi[i]=i-1;
rep(j,1,prime[0]) {
int x=prime[j];
if(i*x>nmax) break;
vis[i*x]=true;
if(i%x==0){
phi[i*x]=phi[i]*x;break;
}else phi[i*x]=phi[i]*phi[x];
}
}
}

poj2407:求欧拉函数

#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
#define rep(i,s,t) for(int i=s;i<=t;i++)
#define clr(x,c) memset(x,c,sizeof(x))
int read(){
int x=0;char c=getchar();
while(!isdigit(c)) c=getchar();
while(isdigit(c)) x=x*10+c-'0',c=getchar();
return x;
}
int getphi(int x){
int ans=x;
for(int i=2;i*i<=x;i++){
if(x%i==0) ans=ans/i*(i-1);
while(x%i==0) x/=i;
}
if(x!=1) ans=ans/x*(x-1);
return ans;
}
int main(){
//rep(i,2,100) printf("%d:%d\n",i,getphi(i));
while(1){
int n=read();
if(!n) break;
printf("%d\n",getphi(n));
}
return 0;
}

poj2478:求欧拉函数前缀和

#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
#define rep(i,s,t) for(int i=s;i<=t;i++)
#define clr(x,c) memset(x,c,sizeof(x))
#define ll long long
int read(){
int x=0;char c=getchar();
while(!isdigit(c)) c=getchar();
while(isdigit(c)) x=x*10+c-'0',c=getchar();
return x;
}
const int nmax=1000005;
int prime[nmax],phi[nmax];
bool vis[nmax];
void getphi(){
phi[1]=1;
rep(i,2,nmax){
if(!vis[i]) prime[++prime[0]]=i,phi[i]=i-1;
rep(j,1,prime[0]) {
int x=prime[j];
if(i*x>nmax) break;
vis[i*x]=true;
if(i%x==0){
phi[i*x]=phi[i]*x;break;
}else phi[i*x]=phi[i]*phi[x];
}
}
}
int main(){
getphi();
//rep(i,1,100) printf("%d:%d\n",i,phi[i]);printf("\n");
while(1){
int n=read();
if(!n) break;
ll ans=0;
rep(i,2,n) ans+=phi[i];
printf("%lld\n",ans);
}
return 0;
}