快速幂
快速幂的作用:
就是为了快速的算出,先看朴素算法,如果算a^k那么得用一个for循环,效率为,但是如果使用快速幂那么效率就成变成了,所以说效率还是很高的。
原理:
因为k有个二进制位,所以我们需要预处理出, ,.....,一共有个,因此时间复杂度是,然后再组合出即可,就是将拆成的形式,整理一下就是的形式,因此我们可以分两步:
- 第一步先处理k的二进制:假设,那么k的二进制形式就是,因此二进制位上为1的加上即可
- 第二步:快速处理出, ....,很容易发现,,即每一项是前一项的平方
都处理完之后,就可以把拆成的形式了
感觉自己说的都一懵一懵的,举个例子:
假设要算,那么我们可以将拆解成它的二进制形式,于是:
代码:
int qmi(int a, int k, int p) {
int res = 1;
if(k == 0 && p == 1) return 0; //可能会出现k == 0 p == 1的情况, 需要特判
while(k) {
if(k & 1) res = (LL)res * a % p; //不要忘了long long很重要
k >>= 1;
a = (LL)a * a % p;
}
return res;
}
矩阵快速幂
与普通的快速幂类似,求(A是一个矩阵)只不过是多了一个矩阵乘法,再有就是注意一些细节了,long long的使用,直接看代码吧
#include <iostream>
#include <cstdio>
#include <cstring>
using namespace std;
typedef long long LL;
const int N = 110, Mod = 1e9 + 7;
long long n, k;
void mul(LL a[][N], LL b[][N], LL c[][N]) {
LL temp[N][N] = {0};
for (int i = 0; i < n; i++) {
for (int j = 0; j < n; j++) {
for (int k = 0; k < n; k++) {
temp[i][j] = (temp[i][j] + (LL)b[i][k] * c[k][j] % Mod) % Mod; //防止溢出
}
}
}
memcpy(a, temp, sizeof temp);
}
int main() {
LL a[N][N], res[N][N];
scanf("%lld%lld", &n, &k); //因为都是long long类型 因此用lld读入
for (int i = 0; i < n; i++) {
for (int j = 0; j < n; j++) {
scanf("%lld", &a[i][j]);
res[i][j] = a[i][j];
}
}
k--;
while(k) {
if(k & 1) mul(res, res, a);
mul(a, a, a);
k >>= 1;
}
for (int i = 0; i < n; i++) {
for (int j = 0; j < n; j++) {
printf("%lld ", res[i][j]);
}
puts("");
}
return 0;
}
还有一些典型的例题 比如fib求值,求前n项和等等,会填坑的
分类:
之前的比赛题目 / 算法从0开始
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· AI与.NET技术实操系列:基于图像分类模型对图像进行分类
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 25岁的心里话
· 闲置电脑爆改个人服务器(超详细) #公网映射 #Vmware虚拟网络编辑器
· 零经验选手,Compose 一天开发一款小游戏!
· 因为Apifox不支持离线,我果断选择了Apipost!
· 通过 API 将Deepseek响应流式内容输出到前端