NYOJ--102--次方求模(快速求幂取模)

次方求模

时间限制:1000 ms  |  内存限制:65535 KB

难度:3

描述

求a的b次方对c取余的值

输入
第一行输入一个整数n表示测试数据的组数(n<100)
每组测试只有一行,其中有三个正整数a,b,c(1=<a,b,c<=1000000000)
输出
输出a的b次方对c取余之后的结果
样例输入
3
2 3 5
3 100 10
11 12345 12345
样例输出
3
1
10481
 1 /*
 2     Name: NYOJ--102--次方求模
 3     Copyright: ?2017 日天大帝
 4     Author: 日天大帝
 5     Date: 28/04/17 20:31
 6     Description: 快速求幂取模
 7 */
 8 #include<iostream>
 9 using namespace std;
10 const int MAX = 1e8;
11 int pow(long long int n,long long int p,long long int c) {
12     int m = 1;
13     while(p > 0){
14         if(p & 1)
15             m = ((m%c) * (n%c)) %c;
16         n = ((n%c)*(n%c)) %c;
17         p = p>>1;
18     }
19     return m;
20 }
21 int main(){
22     ios::sync_with_stdio(false);
23     int n;cin>>n;
24     while(n--)    {
25         int a,b,c;
26         cin>>a>>b>>c;
27         cout<<pow(a,b,c)<<endl;
28     }
29     return 0;
30 }

公式求幂→二分求幂→快速求幂→快速求幂取模

直接用C语言的库函数pow()(别忘了它的头文件#include<math.h>),似乎很简单,但是它的时间复杂度高达O(n)。 
显然,这很容易超时。 
于是有了下面的二分求幂(时间复杂度O(lgn))

二分求幂的原理可以用下面这张图表示 

用递归来实现,虽然代码有点长,但是很好理解

 1 int pow(int a,int n)//返回值是a的n次方
 2 {
 3     if(n==0)//递归终止条件
 4         return 1;
 5     if(n==1)
 6         return a;
 7     int result=pow(a,n/2);//二分递归
 8     result=result*result;//这部分奇数偶数都一样
 9     if(n%2==1)//如果n是奇数,就要多乘一次
10         result=result*a;
11     return result;
12 }

用非递归,更加简洁

 1 int pow(int a,int n)//返回值是a的n次方
 2 {
 3     int result=1;
 4     while(n!=0)
 5     {
 6         if(n%2==1)//如果n是奇数
 7             result=result*a;//就要多乘一次
 8         a=a*a;
 9         n=n/2;//二分
10     }
11     return result;
12 }

快速幂顾名思义比二分幂又快一些, 
快速幂借助了强大的位运算,时间复杂度达到O(log?N)。 
用非递归的代码实现

 1 int pow(int a,int n)//返回值是a的n次方
 2 {
 3     int result=1,flag=a;
 4     while(n!=0)
 5     {
 6         if(n&1)//如果n是奇数,即n的二进制最末位为1时
 7             result=result*flag;
 8         flag=flag*flag;
 9         n=n>>1;//n的二进制右移一位,即n/2
10     }
11     return result;
12 }

当然还能用递归来实现,但是太复杂,我没学会…

刷题中让直接求幂的不多,求幂后取模的却不少,毕竟求幂结果太大了。 
水平所限,只会用二分幂取模,时间复杂度与二分幂一样O(lgn)。 
基本可以在各种比赛中顺利通过,也是目前比较常用的方法

原理同样很简单,都是小学学过的:积的取余等于取余的积取余 
接下来用代码实现

 1 int pow(int a,int n,int b)//返回值是a的n次方对b取余后的值
 2 {
 3     int result=1;
 4     a=a%b;//积的取余等于取余的积取余
 5     while(n>0)
 6     {
 7         if(n%2==1)
 8             result=result*a%b;//n是奇数的话就要多乘一次,原理和前面的二分求幂一样
 9         n=n/2;//二分
10         a=a*a%b;//积的取余等于取余的积取余
11     }
12     return result;
13 }

影响计算机效率的是运算次数,而不是运算结果。 
所以前面几个算法都是通过增大运算结果,减少运算次数,提高计算机效率。

时间: 2024-10-12 19:02:14

NYOJ--102--次方求模(快速求幂取模)的相关文章

NYOJ 102 次方求模

次方求模 时间限制:1000 ms  |  内存限制:65535 KB 难度:3 描述 求a的b次方对c取余的值 输入 第一行输入一个整数n表示测试数据的组数(n<100) 每组测试只有一行,其中有三个正整数a,b,c(1=<a,b,c<=1000000000) 输出 输出a的b次方对c取余之后的结果 样例输入 3 2 3 5 3 100 10 11 12345 12345 样例输出 3 1 10481 算法分析: 大数问题,需要利用快速幂取模算法. 所谓的快速幂,实际上是快速幂取模的缩

hdu 3221 Brute-force Algorithm(快速幂取模,矩阵快速幂求fib)

http://acm.hdu.edu.cn/showproblem.php?pid=3221 一晚上搞出来这么一道题..Mark. 给出这么一个程序,问funny函数调用了多少次. 我们定义数组为所求:f[1] = a,f[2] = b, f[3] = f[2]*f[3]......f[n] = f[n-1]*f[n-2].对应的值表示也可为a^1*b^0%p,a^0*b^1%p,a^1*b^1%p,.....a^fib[n-3]*b^fib[n-2]%p.即a,b的指数从n=3以后与fib数列

The 2018 ACM-ICPC China JiangSu Provincial Programming Contest快速幂取模及求逆元

题目来源 The 2018 ACM-ICPC China JiangSu Provincial Programming Contest 35.4% 1000ms 65536K Persona5 Persona5 is a famous video game. In the game, you are going to build relationship with your friends. You have N friends and each friends have his upper b

UVA - 11029Leading and Trailing(快速幂取模取后三位 + log10()取前三位)

题目: UVA - 11029Leading and Trailing(快速幂取模取后三位 + log10()取前三位) 题目大意:给你N的k次方,然后要求你求出这个数的前三位和后三位. 解题思路:因为n和k都很大,这个数求出来是大数,所以可以用快速幂取模求后三位,因为后面的三位和前面的位数的没有关系.前面的三位比较难办.设x = log (n^k)  = k * log10(n),那么10^x = k * log10(n).将X = a(整数) + b(小数),整数部分10^a只是移动小数点,

HDU 5895 矩阵快速幂+高次幂取模

HDU 5895 Mathematician QSC 题意:已知f(n)=2*f(n-1)+f(n-2), g(n)=∑f(i)²(0<=i<=n), 给出n,x,y,s, 求x^(g(n*y))%(s+1); 思路:OEIS查到了g(n)=f(n)*f(n+1)/2, f(n)可以用矩阵快速幂求得, 有一个定理可以用于高次幂取模 x^n %k=x^(n%phi(k)+phi(k)) %k, 此处phi(x)为欧拉函数,但是在对幂次取模时存在一个除2, 又因为(a/b)%k=(a%bk)/b,

快速幂取模算法

什么是快速幂? 快速幂应当是快速幂取模的简称 对于一般的求幂算法,求$a^b\,\bmod\,m$,即使用循环b次的方法,复杂度是$O(b)$的,当b很大的时候,这种算法就会显得十分缓慢. 快速幂是基于以下明显的事实: $${a^b} \equiv {(a^2)^{\frac{b}{2}}} \pmod{m}\quad b\ is\ even$$ $${a^b} \equiv {(a^2)^{\frac{b}{2}}*a} \pmod{m}\quad b\ is\ odd$$ 那么我们得到这样一

快速幂取模

参考文章来源:Reait  Home(http://www.reait.com/blog.html) 转载请注明,谢谢合作. 在Miller Rabbin测试素数,就用到了快速幂取模的思想.这里总结下.求a^b%c(这就是著名的RSA公钥的加密方法),当a,b很大时,直接求解这个问题不太可能 算法1:利用公式a*b%c=((a%c)*b)%c,这样每一步都进行这种处理,这就解决了a^b可能太大存不下的问题,但这个算法的时间复杂度依然没有得到优化 代码如下: 01.int modexp_simpl

快速幂取模和快乘取模

一.快速幂取模概念 快速幂取模,顾名思义,就是快速的求一个幂式的模(余),比如a^b%c,快速的计算出这个式子的值. 在程序设计过程中,经常要去求一些大数对于某个数的余数,为了得到更快.计算范围更大的算法,产生了快速幂取模算法. 二.快速幂取模算法实现 1)很容易能想到,循环b次,每次乘a,最后对c取余就可以了. int ans = 1; for(int i = 1; i<=b; i++) { ans = ans * a; } ans = ans % c; 这个朴素算法的问题是: 1.如果a和b

1046 A^B Mod C(快速幂取模)

1046 A^B Mod C(51NOD基础题) 基准时间限制:1 秒 空间限制:131072 KB 分值: 0 难度:基础题 给出3个正整数A B C,求A^B Mod C. 例如,3 5 8,3^5 Mod 8 = 3. Input 3个正整数A B C,中间用空格分隔.(1 <= A,B,C <= 10^9) Output 输出计算结果 Input示例 3 5 8 Output示例 3 /* 1046 A^B Mod C(快速幂取模) 给出3个正整数A B C,求A^B Mod C. (