标签:select 基于 位置 back 原理 href htm src ++
1 bool prime(int q) 2 { 3 if(q==2||q==3) return 1; 4 if(q==1) return 0; 5 if(q%6!=1||q%6!=5) return 0; 6 int cnt=sqrt(q); 7 for(int i=5;i<=cnt;i+=6) 8 if(q%2!=0||q%(i+2)!=0) return 0; 9 return 1; 10 }
1 //埃氏筛 筛出1~n的素数 2 void prime_select() 3 { 4 for(int i=2;i<=n;i++) 5 { 6 if(vis[i]) continue; 7 printf("%d\n",i); 8 for(int j=n;j<=n/i;j++) vis[i*j]=1; 9 } 10 }
线性筛还是要学的qwq(真香),它的原理是每个合数会被它的最小质因子筛一次,利用了当前已经筛出的质数。复杂度真·O(N)
1 //线性筛 2 void prime_select() 3 { 4 //v[]记录下标数的最小质因子 初值为0 5 for(int i=2;i<=n;i++) 6 { 7 if(v[i]==0) v[i]=i,prime[++m]=i; 8 for(int j=1;j<=m;j++) 9 {//i是比prime[j]更小的质因子or超出n的范围 10 if(prime[j]>v[i]||prime[j]>n/i) break; 11 v[i*prime[j]]=prime[j]; 12 } 13 } 14 }
1 //质因数分解--基于算术基本定理 复杂度O(根号n) 2 void divide() 3 { 4 for(int i=2;i<=sqrt(n);i++) 5 if(n%i==0) 6 { 7 p[++m]=i;c[m]=0; 8 while(n%i==0) n/=i,c[m]++; 9 } 10 if(n>1) p[++m]=n,c[m]=1; 11 for(int i=1;i<=m;i++) 12 printf("%d^%d\n",p[i],c[i]); 13 }
丢几个例题跑嘤嘤嘤
例题1 LuoguP1865 A%B Problem ---(本博客开通不久的旧文)
因为数据范围较水,仅1e6,所以我们可以先使用线性筛筛出素数。区间个数用前缀和维护。它珂以当做一个练线性筛的不错模板题。
例题2 UVA10140 Prime Distance --(题解一篇)
我们知道,任意一个合数x一定包含不超过sqrt(n)的质因子。
所以我们就筛出2~sqrt(R)之间的所有素数,用他们来标记全部范围内的合数。最后没被标记的数就是质数,比较相邻的质数位置取最大。
例题3 阶乘分解 没有题面,口胡一下。
把N!分解质因数,按算术基本定理的形式输出。(N为1e6级别)
N!中质因数p的个数就等于1~N每个数含质因子p的个数之和。其他...详见lyd书p131,不会用LaTex,懒得打了...
时间复杂度O(Nlogn)
// 这样写书式的复习我肯定干不完...以后会简洁一点...(真香)
//求1~N每个数的正约数集合--倍数法 void work() { vector<int>fac[500010]; for(int i=1;i<=n;i++) for(int j=1;j<=n/i;j++) fac[i*j].push_back(i); }
复杂度为O(N+N/2+N/3+N/4+...+N/N)=O(NlogN)(调和级数)
例题1 LuoguP1463反素数
例题2 LuoguP2261余数之和
int gcd(int a,int b) { return b ? gcd(b,a%b) : a; }//辗转相除 int gcd(int a,int b) { while(a!=b) { if(a>b) a-=b; else b-=a; } return a; }//更相减损
void phi() { phi[1]=1; for(int i=2;i<=n;i++) phi[i]=i; for(int i=2;i<=n;i++) if(phi[i]==i) for(int j=i;j<=n;j+=i) phi[j]=phi[j]/i*(i-1); }
标签:select 基于 位置 back 原理 href htm src ++
原文地址:https://www.cnblogs.com/nopartyfoucaodong/p/9697667.html