zoukankan      html  css  js  c++  java
  • POJ 3641 Pseudoprime numbers (快速幂模)

    题目链接http://poj.org/problem?id=3641 题目大意:给定a,p,判断一个数是不是伪素数(伪素数是指p---p满足a^p = a (mod p)但p本身不是素数),2<=a,p<=1000000000. 思路: 我们发现指数p很大,不过因为结果mod p并且由定理(a * a) % p = (a % p) * (a % p)可知结果在int范围内,但是暴力解a^p显然还是不可能的. 这里就要用到一个著名的算法:蒙哥马利模算法(快速幂模)来快速计算a^p % m: 这种算法利用了二分的思想,可以达到O(logn)。 可以把b按二进制展开为:b = p(n)*2^n  +  p(n-1)*2^(n-1)  +…+   p(1)*2  +  p(0)(其中p(i) (0<=i<=n)为 0 或 1) 这样 a^b =   a^(p(n)*2^n)  *  a^(p(n-1)*2^(n-1))  *...*  a^(p(1)*2)  *  a^p(0) 对于p(i)=0的情况, a^(p(i) * 2^(i-1) ) = 1不用处理.我们要考虑的仅仅是p(i)=1的情况: 化简:a^(2^i)  = (  a^(  2^(i-1)  )  )^2 (这里很重要!!具体请参阅秦九韶算法:http://zh.wikipedia.org/wiki/%E7%A7%A6%E4%B9%9D%E9%9F%B6%E7%AE%97%E6%B3%95) 利用这一点,我们可以递推地算出所有的a^(2^i) 当然由算法1的结论,我们加上取模运算:a^(2^i)%c = ( (a^(2^(i-1))%c) * a^(2^(i-1)))  %c 于是再把所有满足p(i)=1的a^(2^i)%c按照算法1乘起来再%c就是结果, 即二进制扫描从最高位一直扫描到最低位. 扩展当a为一个矩阵时,算法就是矩阵快速幂.   递归代码(我一直觉得递归真的很优美~~~):
    int exp_mod(int a, int p, int m){   //a^p (mod m)
        if (p == 0)
            return 1;
        if (p == 1)
            return a % m;
        
    	long long t = exp_mod(a, p/2, m);	//long long防止下面t*t时溢出
        
    	t = t * t % m;
        
    	if (p & 1)
            t = t * a % m;
        
    	return t;
    }
    
      非递归代码:  
    //计算a^bmodn   
    int modexp(int a, int b, int n)   
    {   
        int ret = 1;   
        long long tmp = a;   
        while(b)   
        {   
           //基数存在   
           if (b & 1) ret = ret * tmp % n;   
           tmp = tmp * tmp % n;   
           b >>= 1;   
        }   
        return ret;   
    }     
    
      POJ 3641代码:  
    #include 
    using namespace std;
    bool prime(int n){
        for (int i = 2; i * i <= n; i ++){
            if (n % i == 0){
                return false;
            }
        }
        return true;
    }
    int exp_mod(int a, int p, int m){   //a^p (mod m)
        if (p == 0)
            return 1;
        if (p == 1)
            return a % m;
        long long t = exp_mod(a, p/2, m);
        t = t * t % m;
        if (p & 1)
            t = t * a % m;
        return t;
    }
    int main(){
        int p, a;
        while(cin >> p >> a){
            if (p + a == 0){
                return 0;
            }
            if (exp_mod(a, p, p) == a){
                if (!prime(p)){
                    cout << "yes\n";
                }
                else{
                    cout << "no\n";
                }
            }
            else{
                cout << "no\n";
            }
        }
        return 0;
    }
    
     
    举杯独醉,饮罢飞雪,茫然又一年岁。 ------AbandonZHANG
  • 相关阅读:
    apue学习笔记(第一章UNIX基础知识)
    批处理之发布新版本
    在Vista或Windows 7系统上安装Sharepoint 2007
    SharePoint Server 2007 简体中文下载
    sql连接字符串的方法
    共享本地的无线网络
    FastReport报表
    C# 语音识别(文字to语音、语音to文字)
    C#调用脚本语言(三)-- IronJS 与 IronLua 简单方法性能比较
    VS2010中出现无法嵌入互操作类型
  • 原文地址:https://www.cnblogs.com/AbandonZHANG/p/4113980.html
Copyright © 2011-2022 走看看