Uva - 10006 - Carmichael Numbers

题意:判断一个数n(2 < n < 6500)是否是一个Carmichael数,

成立条件:1、是合数;

2、对于任意一个2<= a <= n-1,都是有a^n mod n = a。

题目链接:http://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem&category=12&problem=947

——>>刚学了个矩阵快速幂,想起还有个快速幂取模,找到这题,核心思想与矩阵快速幂是一样的,另外加上个合数的判断,判断方法可先打素数表,二分查找。

另外有个重大发现,那个mod的n是不变的(就是输入的n),而那个指数n是变化的。

#include <cstdio>
#include <cstring>

using namespace std;

const int maxn = 65000 + 10;
int p[maxn], cnt;
bool vis[maxn];

void prime_table()      //筛素数
{
    memset(vis, 0, sizeof(vis));
    for(int i = 2; i < maxn; i++) if(!vis[i])
        for(int j = (i<<1); j < maxn; j += i) vis[j] = 1;
    cnt = 0;
    for(int i = 2; i < maxn; i++)
        if(!vis[i]) p[cnt++] = i;
}
bool bs(int* A, int x, int y, int v)        //二分查找
{
    int m;
    while(x < y)
    {
        m = x + (y-x) / 2;
        if(A[m] == v) return 1;
        else if(A[m] > v) y = m;
        else x = m+1;
    }
    return 0;
}
int pow_mod(int a, int n, int m)        //二分幂取模
{
    if(n == 1) return a % m;
    long long x = pow_mod(a, n/2, m);
    long long ans = x * x % m;
    if(n&1) ans = ans * a % m;
    return ans;
}
int main()
{
    int n;
    prime_table();
    while(~scanf("%d", &n))
    {
        if(!n) return 0;
        bool ok = 0;
        if(!bs(p, 0, cnt, n))
        {
            ok = 1;
            for(int i = 2; i <= n-1; i++)
                if(pow_mod(i, n, n) != i)
                {
                    ok = 0;
                    break;
                }
        }
        if(ok) printf("The number %d is a Carmichael number.\n", n);
        else printf("%d is normal.\n", n);
    }
    return 0;
}


你可能感兴趣的:(Uva - 10006 - Carmichael Numbers)