2018.12.17 hdu2138 How many prime numbers(miller-rbin)

传送门

miller−rabbinmiller-rabbinmiller−rabbin素数测试的模板题。

实际上miller−rabinmiller-rabinmiller−rabin就是利用费马小定理和二次探测的性质来进行判断的。

注意要多带几个素数进去判断(听大神说只要取遍了50以内的质数就可以判断intintint范围内的)

代码:

#include<bits/stdc++.h>
#define ri register int
using namespace std;
inline int read(){
	int ans=0;
	char ch=getchar();
	while(!isdigit(ch))ch=getchar();
	while(isdigit(ch))ans=(ans<<3)+(ans<<1)+(ch^48),ch=getchar();
	return ans;
}
typedef long long ll;
int s,t,n,ans,pri[14]={2,3,5,7,11,13,17,19,23,29,31,37,41,47};
inline int ksm(int a,int p,int mod){int ret=1;for(;p;p>>=1,a=(ll)a*a%mod)if(p&1)ret=(ll)ret*a%mod;return ret;}
inline bool check(int x,int a){
	if(ksm(a,x-1,x)^1)return 0;
	a=ksm(a,t,x);
	if(a==1||a==x-1)return 1;
	for(ri i=0;i<s;++i){
		a=(ll)a*a%x;
		if(a==x-1)return 1;
	}
	return 0;
}
inline bool MRT(int x){
	if(!(x&1))return x==2;
	s=0,t=x-1;
	while(!(t&1))t>>=1,++s;
	for(ri i=0;i<14;++i){
		if(x==pri[i])return 1;
		if(!(x%pri[i]))return 0;
		if(!check(x,pri[i]))return 0;
	}
	return 1;
}
int main(){
	freopen("lx.in","r",stdin);
	while(~scanf("%d",&n)){
		ans=0;
		while(n--)if(MRT(read()))++ans;
		cout<<ans<<'\n';
	}
	return 0;
}
上一篇:2018.12.17 bzoj4802: 欧拉函数(Pollard-rho)


下一篇:Yii入门☞应用