标签:void name 质数 sub namespace 标记 oid 记录 str
费马定理:
ap≡a(mod p)
其中p为质数,且a不是p的倍数
证明:
。。。。。
欧拉定理:
aφ(p)≡1(mod p)
φ(x)(欧拉函数)为小于等于x且与x互质的数的个数
φ(x)=∏(pi-1)*piki-1 其中pi表示 x的质因数,ki表示这种质因数的个数
特别的对于质数 φ(x)=x-1。
欧拉函数的代码实现:
1 #include<cstdio> 2 #include<Iostream> 3 using namespace std; 4 int ol(int x) 5 { 6 int ans=1; 7 for(int i=2;i*i<=x;++i) 8 { 9 if(x%i==0) 10 { 11 x/=i; 12 ans*=i-1; 13 } 14 while(x%i==0) 15 { 16 x/=i; 17 ans*=i; 18 } 19 } 20 if(x>1) ans*=x-1; 21 return ans; 22 } 23 int main() 24 { 25 int a; 26 scanf("%d",&a); 27 printf("%d",ol(a)); 28 return 0; 29 }
最后函数里那个如果x>1,ans*=x-1一开始让我很懵,后来一想,如果这个数将所有的质因数除过一遍之后,剩下的数如果不是1,那么剩下的肯定只有一个并且是个质数(证明很显然)
筛法
有两种筛法,第一种叫做埃拉托斯特尼筛法(复杂度O(n log logn)),另一种是欧拉筛法(复杂度O(n))
埃拉托斯特尼筛法其实就是用已得到质数,去将他的所有n以内倍数标记为合数,最后剩下的就是合数。
在进行筛法的同时,可以顺便找到每个数的最小质因数(就是第一次更新他的那个质数)
欧拉筛法:在埃氏筛法中每一个合数可能会被更新很多遍,这些是没有必要的,所以就有了欧拉筛。
欧拉筛的思路就是保证每个合数只被他的最小质因数筛掉。用一个数组dis来存已经得到的素数,
然后再用已经得到素数取筛其他的数,主要的地方是代码中 if(i%dis[j]==0) break; 这个地方。
两个筛法的代码
1 #include<cstdio> 2 #include<iostream> 3 4 using namespace std; 5 const int N=10000; 6 void aa(int n); 7 void ol(int n); 8 int main() 9 { 10 int n; 11 scanf("%d",&n); 12 aa(n); 13 printf("\n"); 14 ol(n); 15 } 16 17 void aa(int n)//埃氏筛法 18 { 19 int vis[N]; 20 int dis[N];//用来存最小质因数 21 for(int i=2;i<=n;++i) 22 { 23 if(!vis[i]) 24 { 25 for(int j=i*2;j<=n;j+=i) 26 { 27 if(!vis[j]) 28 { 29 vis[j]=1; 30 dis[j]=i; 31 } 32 } 33 } 34 } 35 for(int i=1;i<=n;++i) 36 { 37 if(!vis[i]) 38 { 39 printf("%d ",i); 40 } 41 } 42 } 43 void ol(int n) 44 { 45 int vis[N]; 46 int dis[N],js=0; 47 for(int i=2;i<=n;++i) 48 { 49 if(!vis[i]) 50 { 51 dis[++js]=i; 52 } 53 for(int j=1;dis[j]*i<=n;++j) 54 { 55 vis[i*dis[j]]=1; 56 if(i%dis[j]==0) break; 57 } 58 } 59 for(int i=1;i<=n;++i) 60 { 61 if(!vis[i]) printf("%d ",i); 62 } 63 }
快速幂与快速乘法
尽管快速幂与快速乘法好像扯不上什么关系,但是东西不是很多,就一起整理到这里吧
快速幂思想就是将ax看作x个a相乘,用now记录当前答案,然后将指数每次除以2,然后将当前答案平方,如果x的2进制最后一位为1的话,就将答案乘以现在的数。快速乘法类似,只是将a*x看作x个a相加。
代码
1 #include<cstdio> 2 #include<iostream> 3 using namespace std; 4 int mi(int a,int x) 5 { 6 int ans=1; 7 for(int now=a;x>=1;x>>=1,now=now*now)//a表示底数,x表示次数 8 { 9 if(x&1) ans=ans*now; 10 } 11 return ans; 12 } 13 int cheng(int a,int x)//表示a*x 14 { 15 int ans=0; 16 for(int now=a;x>=1;now=now*2,x>>=1) 17 { 18 if(x&1) ans=ans+now; 19 } 20 return ans; 21 } 22 int main() 23 { 24 int a,x; 25 scanf("%d%d",&a,&x); 26 printf("快速幂 %d\n",mi(a,x)); 27 printf("快速乘法 %d\n",cheng(a,x)); 28 return 0; 29 }
标签:void name 质数 sub namespace 标记 oid 记录 str
原文地址:https://www.cnblogs.com/wxyww/p/8982762.html