Carmichael Numbers (快速幂)

 

    当今计算机科学的一个重要的领域就是密码学。有些人甚至认为密码学是计算机科学中唯一重要的领域,没有密码学生命都没有意义。
  阿尔瓦罗就是这样的一个人,它正在设计一个为西班牙杂烩菜饭加密的步骤。他在加密算法中应用了一些非常大的素数。然而确认一个非常大的数是不是素数并不是那么简单。一个费时的方法是用比这个数的平方根小的所有素数去除它,对于大整数来说,这样一定会毁掉这个杂烩菜饭的。
  然而,一些很有信心耗时少的随机测试存在,其中一个就是费马测试。
  在2和n-1之间随机选取一个数(n是我们要测试的数)。如果a n mod n = a 成立,n就可能是一个素数。
  如果一个数通过费马测试很多次那么它就很可能是一个素数。
  不幸的是,一些数不是素数但是它们依然能通过每一个比它小的数的费马测试。这些数被称作卡迈克尔数
  这道题要求你写一个程序去测试给定的数是不是一个卡迈克尔数。
  完成了这个任务的队伍有希望接受来自阿尔瓦罗的西班牙杂烩菜饭23333
 
 
多组输入,第一行给一个n (2 < n < 65000) 。n = 0 表示输入结束并不需要处理
 
对每组输入,输出它是不是卡迈克尔数,参考样例。
 
 
1729
17
561
1109
431
0
 
 
The number 1729 is a Carmichael number.
17 is normal.
The number 561 is a Carmichael number.
1109 is normal.
431 is normal.
 
 
 
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;

//尚不是特别大的数嘞! 
//打表计算素数
int prime[65000];
void f()
{
	prime[0]=prime[1]=0;
	for(int i=2;i<65000;i++) prime[i]=1;
	for(int i=2;i<65000;i++)
		if(prime[i])
			for(int j=2*i;j<65000;j+=i)
				prime[j]=0;
}

ll qmod(ll a,ll b,ll m)//计算a^b MOD m
{
	ll ans=1;
	while(b)
	{
		if(b&1)  ans=(ans*a)%m;
		b>>=1;
		a=(a*a)%m;
	}
	return ans;
}

int tests(int n)
{
	for(int i=2;i<=n-1;i++)
		if(qmod(i,n,n)!=i) return 0;//点睛巨笔!
	return 1;
}

int main()
{
	f();

	int n;
	while(cin>>n&&n)
	{
		if(!prime[n] && tests(n))
			cout<<"The number "<<n<<" is a Carmichael number.\n";
		else cout<<n<<" is normal.\n";
	}
	return 0;
}

  

 

附录:https://vjudge.net/contest/310908#problem/A

 

 
 
 
 
posted @ 2019-08-14 10:40  龙龙666666  阅读(465)  评论(0编辑  收藏  举报