定义

快速求a^b%c的算法

原理

指数可以被二进制分解

那么a^b可以分解为a^2^k1*a^2^k2*……

又显然a^2^(k+1)=a^(2^k*2)=(a^2^k)^2

所以可以将指数在二进制下从低位向高位递推,每次将底数平方,若该位是1就将答案乘上底数,直到指数为0。

实现时可以每次将指数/2方便处理

位运算优化

x&1:取x二进制下最后一位

x>>1:x/2

代码

int quickpow(int a,int b,const int c)
{
int base=a%c,ans=;
while(b)
{
if(b&)
ans=ans*base%c;
base=base*base%c;
b>>=;
}
return ans;
}

快速幂

例题

一、序列的第k个数

根据元素之差判断是不是等差数列,不是等差数列即为等比数列

推通项公式时注意序列起始为a

#include<cstdio>
#include<cctype>
using namespace std;
#define re register int
#define ll long long
int stk[],tt;
void print(ll x)
{
if(x==)
putchar('');
else
{
if(x<)
putchar('-'),x=-x;
tt=;
while(x)
{
stk[++tt]=x%;
x/=;
}
for(re i=tt;i;i--)
putchar(stk[i]|);
}
}
int read()
{
int x=,f=;
char c=getchar();
while(!isdigit(c))
{
f|=c=='-';
c=getchar();
}
while(isdigit(c))
{
x=(x<<)+(x<<)+(c^);
c=getchar();
}
return f?-x:x;
}
const int MOD=;
ll quickpow(int a,int b,const int c)
{
ll base=a%c,ans=;
while(b)
{
if(b&)
ans=ans*base%c;
base=base*base%c;
b>>=;
}
return ans;
}
int main()
{
int T=read();
ll a,b,c,k;
while(T--)
{
a=read(),b=read(),c=read(),k=read();
if(b-a==c-b)
print((a+(b-a)*(k-))%MOD);
else
print(a*quickpow(b/a,k-,MOD)%MOD);
putchar('\n');
}
return ;
}

序列的第k个数

二、[NOIP2013]转圈游戏

走10^k轮即移动m*10^k个位置,再加上x取模即可

#include<cstdio>
#include<cctype>
using namespace std;
#define re register int
#define ll long long
int stk[],tt;
void print(int x)
{
if(x==)
putchar('');
else
{
if(x<)
putchar('-'),x=-x;
tt=;
while(x)
{
stk[++tt]=x%;
x/=;
}
for(re i=tt;i;i--)
putchar(stk[i]|);
}
}
int read()
{
int x=,f=;
char c=getchar();
while(!isdigit(c))
{
f|=c=='-';
c=getchar();
}
while(isdigit(c))
{
x=(x<<)+(x<<)+(c^);
c=getchar();
}
return f?-x:x;
}
ll quickpow(int a,int b,const int c)
{
ll base=a%c,ans=;
while(b)
{
if(b&)
ans=ans*base%c;
base=base*base%c;
b>>=;
}
return ans;
}
int main()
{
int n=read(),m=read(),k=read(),x=read();
ll ans=(x+m*quickpow(,k,n))%n;
print(ans);
putchar('\n');
return ;
}

转圈游戏

三、[HNOI2008]越狱

可越狱方案数不好求,但从容斥原理的角度,答案可以表示成总排列数-不可越狱排列数

这两个数都很好求,总排列数=m^n,不可越狱排列数考虑第一个数有m种选法,后面每个数都只有m-1种选法,于是=m*(m-1)^(n-1)

#include<cstdio>
#include<cctype>
using namespace std;
#define re register int
#define ll long long
int stk[],tt;
void print(int x)
{
if(x==)
putchar('');
else
{
if(x<)
putchar('-'),x=-x;
tt=;
while(x)
{
stk[++tt]=x%;
x/=;
}
for(re i=tt;i;i--)
putchar(stk[i]|);
}
}
ll read()
{
ll x=;
int f=;
char c=getchar();
while(!isdigit(c))
{
f|=c=='-';
c=getchar();
}
while(isdigit(c))
{
x=(x<<)+(x<<)+(c^);
c=getchar();
}
return f?-x:x;
}
const int MOD=;
ll quickpow(int a,ll b)
{
ll base=a%MOD,ans=;
while(b)
{
if(b&)
ans=ans*base%MOD;
base=base*base%MOD;
b>>=;
}
return ans;
}
inline int mod(ll a)
{
a%=MOD;
if(a<)
a+=MOD;
return a;
}
int main()
{
ll m=read(),n=read();
print(mod(quickpow(m,n)-m*quickpow(m-,n-)));
putchar('\n');
return ;
}

越狱

注意事项

1、根据题目数据范围适当修改快速幂函数中数据的类型

2、取模的数可能<0时注意实际意义

快速幂(Fast Pow)的更多相关文章

  1. hdu 1757 矩阵快速幂 **

    一看正确率这么高,以为是水题可以爽一发,结果是没怎么用过的矩阵快速幂,233 题解链接:点我 #include<iostream> #include<cstring> ; us ...

  2. HUST 1569(Burnside定理+容斥+数位dp+矩阵快速幂)

    传送门:Gift 题意:由n(n<=1e9)个珍珠构成的项链,珍珠包含幸运数字(有且仅由4或7组成),取区间[L,R]内的数字,相邻的数字不能相同,且旋转得到的相同的数列为一种,为最终能构成多少 ...

  3. A Simple Math Problem(矩阵快速幂)(寒假闭关第一题,有点曲折啊)

    A Simple Math Problem Time Limit: 3000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Other ...

  4. hdu4965 Fast Matrix Calculation (矩阵快速幂 结合律

    http://acm.hdu.edu.cn/showproblem.php?pid=4965 2014 Multi-University Training Contest 9 1006 Fast Ma ...

  5. LeetCode 50 - Pow(x, n) - [快速幂]

    实现 pow(x, n) ,即计算 x 的 n 次幂函数. 示例 1: 输入: 2.00000, 10输出: 1024.00000 示例 2: 输入: 2.10000, 3输出: 9.26100 示例 ...

  6. LeetCode Pow(x, n) (快速幂)

    题意 Implement pow(x, n). 求X的N次方. 解法 用正常的办法来做是会超时的,因为可能有21亿次方的情况,所以需要优化一下.这里用到了快速幂算法,简单来说就是将指数分解成二进制的形 ...

  7. HDU4965 Fast Matrix Calculation —— 矩阵乘法、快速幂

    题目链接:https://vjudge.net/problem/HDU-4965 Fast Matrix Calculation Time Limit: 2000/1000 MS (Java/Othe ...

  8. hdu4965 Fast Matrix Calculation 矩阵快速幂

    One day, Alice and Bob felt bored again, Bob knows Alice is a girl who loves math and is just learni ...

  9. HDU - 4965 Fast Matrix Calculation 【矩阵快速幂】

    题目链接 http://acm.hdu.edu.cn/showproblem.php?pid=4965 题意 给出两个矩阵 一个A: n * k 一个B: k * n C = A * B M = (A ...

随机推荐

  1. 正则表达式分组(Grouping)

    一 捕获型 (x) 匹配 x ,并且捕获匹配项 const regExp = /(\w+)\s+(\d+)/; const str = 'Android 8'; str.replace(regExp, ...

  2. CSS3——背景 文本 字体 链接 列表样式 表格

    背景 background-color rgb(255,0,0,1)      最后一个值表示透明度,范围是 0--1 background-image 默认平铺重复显示 background-rep ...

  3. 关于migration build failed的问题

    首先一定要执行dotnet restore 查看网站的依赖关系(有时候生成是不报错的但是restore会找不到文件路径) 检查执行命令的路径是否是正确的当前网站路径 build failed一定是生成 ...

  4. Linux-定时任务-打包与压缩

    figure:first-child { margin-top: -20px; } #write ol, #write ul { position: relative; } img { max-wid ...

  5. Double类型的数值 在写入excel时 如何去掉 科学计算法的 后面数值+ E的 情况

    Double start = 20190724100000.000; 写入excel时 是 201907E+13 但想要输出的是 20190724100000 这种格式 Java在java.math包 ...

  6. https=http+ssl

    TLS/SSL中使用了非对称加密,对称加密以及HASH算法.握手过程的具体描述如下: 浏览器将自己支持的一套加密规则发送给网站. 网站从中选出一组加密算法与HASH算法,并将自己的身份信息以证书的形式 ...

  7. Redis进阶:Redis的主从复制机制

    Redis进阶:Redis的主从复制机制 主从复制机制介绍 单机版的Redis存在性能瓶颈,Redis通过提高主从复制实现读写分离,提高了了Redis的可用性,另一方便也能实现数据在多个Redis直接 ...

  8. 用adb logcat抓取log

    实时打印的主要有:logcat main,logcat radio,logcat events,tcpdump,还有高通平台的还会有QXDM日志     状态信息的有:adb shell dmesg, ...

  9. Android使用adb抓完整Log

    前言     最新项目里一直在做 Android RIL 方面的研究,非常最终项目还是未能解决通信底层模块的问题,但是在使用adb抓log上还是有一些收获的,这里记录一下.   Log分类     A ...

  10. Thinkphp3.2 Redis缓存session

    Thinkphpsession缓存没有redis类库 Redis.class.php放在Library/Think/Session/Driver/下: <?php /** * +-------- ...