我是靠谱客的博主 寒冷蓝天,最近开发中收集的这篇文章主要介绍数论(五)—— 快速幂,觉得挺不错的,现在分享给大家,希望可以做个参考。

概述

快速幂用于快速求出 ak mod p 的结果(时间复杂度O(logk),其中 1 ≤ a,p,k ≤ 109
如果不用快速幂的话,就得循环做 a * a 共 k 次,如果 k = 1e9,那么就要乘109次,时间复杂度就太高了,所以我们要用快速幂解决此类问题。
原理
1)先预处理这些值:a20 mod p,a21 mod p,a22 mod p … … a2logk mod p
2)组合出aK也就是把k拆成20,21…若干个2的次幂之和
ak = a2x1·a2x2·… · a2xt = a2x1+2x2+…+2xt
3)把k换算成二进制表示
如 (k)10 = (110110)2
k = 21 + 22 + 24 + 25
举个简单的例子
在这里插入图片描述
模板

#include<iostream>
#include<algorithm>
using namespace std;
typedef long long LL;
// a^k % p
int qmi(int a,int k,int p)
{
int res = 1;
while(k)
{
//k的二进制表示中,末位是1
if(k & 1) res = (LL)res * a % p;
k >>= 1; // k的二进制右移一位
a = (LL)a * a % p; // a不断平方进行更新 
}
return res;
}

应用
快速幂求逆元
乘法逆元的定义
若整数 b,m 互质,并且对于任意的整数 a,如果满足 b|a,则存在一个整数 x,使得 a/b≡a×x(mod m),则称 x 为 b 的模 m 乘法逆元,记为 b−1(mod m)。
b 存在乘法逆元的充要条件是 b 与模数 m 互质。当模数 m 为质数时,bm−2 即为 b 的乘法逆元。
求解方法
1)已知,a/b≡a×x(mod m)

2)两边同乘 b 得,a ≡ a× b × x(mod m)

3)又因为 b 与 m互质,那么 a 一定也与 m 互质,所以,左右两边可以消掉 a,得到 1 ≡ b × x(mod m)

4)即,b × x ≡ 1(mod m),b × b-1 ≡ 1(mod m),简单地说,求逆元就是找到一个x,使其满足式子:b × x ≡ 1(mod m)

5)由费马小定理得知,当 m 为质数时,有 b(m-1) ≡ 1(mod m)

6)将 b 拆开,b * b(m-2) = 1(mod m)

7)对比 4) 和 5) ,可知,x = b(m-2)

所以,求 b模m 的逆元即为求b(m-2)

b(m-2)我们可以借助快速幂来求。
当 b 与 m 不互质时无解,题中已知 m 为质数,那么 b 是 m 的倍数时,b×x也一定时m的倍数,mod m 等于 0 ,此时无解,即b%m == 0 时无解。

给定 n 组 ai,pi,其中 pi 是质数,求 ai 模 pi 的乘法逆元,若逆元不存在则输出 impossible。
注意:请返回在 0∼p−1 之间的逆元。

#include<iostream>
#include<algorithm>
using namespace std;
typedef long long LL;
// a^k % p
int qmi(int a,int k,int p)
{
int res = 1;
while(k)
{
//k的二进制表示中,末位是1
if(k & 1) res = (LL)res * a % p;
k >>= 1; // k的二进制右移一位
a = (LL)a * a % p; // a不断平方进行更新 
}
return res;
}
int main()
{
int n;
scanf("%d",&n);
while(n -- )
{
int a,p;
scanf("%d %d",&a,&p);
int res = qmi(a,p-2,p);
if(a % p != 0) printf("%dn",res);
else puts("impossible");
}
return 0;
}

最后

以上就是寒冷蓝天为你收集整理的数论(五)—— 快速幂的全部内容,希望文章能够帮你解决数论(五)—— 快速幂所遇到的程序开发问题。

如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(40)

评论列表共有 0 条评论

立即
投稿
返回
顶部