专栏首页用户6093955的专栏快速幂和矩阵快速幂(待进一步补充)

快速幂和矩阵快速幂(待进一步补充)

快速幂

如计算 a^b^ ,代码如下:

快速幂代码

快速幂取模:

int multi(int a,int b, int mod)
{
    int ans = 1,base = a;
    while(b!=0)
    {
        if(b&1)
            ans = ans * base % mod;
        base = base * base % mod;
        b>>=1;
    }
    return ans % mod;        //最后不要忘记还要取模
}

快速幂:

int multi(int a,int b, int mod)
{
    int ans = 1,base = a;
    while(b!=0)
    {
        if(b&1)
            ans = ans * base ;
        base = base * base;
        b>>=1;
    }
    return ans;      
}

矩阵快速幂

它可以快速求出斐波那契数列,这里以一个题为例,Fibonacci POJ - 3070

AC代码如下:

#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
using namespace std;
 
const int MOD = 10000;
 
struct matrix {     //矩阵 
    int m[2][2];
}ans;
 
matrix base = {1, 1, 1, 0}; 
 
matrix multi(matrix a, matrix b) {  //矩阵相乘,返回一个矩阵 
    matrix tmp;
    for(int i = 0; i < 2; i++) {
        for(int j = 0; j < 2; j++) {
            tmp.m[i][j] = 0;
            for(int k = 0;  k < 2; k++)
                tmp.m[i][j] = (tmp.m[i][j] + a.m[i][k] * b.m[k][j]) % MOD;
        }
    }
    return tmp;
}
 
int matrix_pow(matrix a, int n) {   //矩阵快速幂,矩阵a的n次幂 
    ans.m[0][0] = ans.m[1][1] = 1;  //初始化为单位矩阵 
    ans.m[0][1] = ans.m[1][0] = 0;
    while(n) {
        if(n & 1) ans = multi(ans, a);
        a = multi(a, a);
        n >>= 1;
    }
    return ans.m[0][1];
}
 
int main() {
    int n;
    // freopen("in.txt", "r", stdin);
    // freopen("out.txt", "w", stdout);
    while(scanf("%d", &n), n != -1) {
        printf("%d\n", matrix_pow(base, n));
    }
    return 0;
} 

参考博客:https://blog.csdn.net/alps1992/article/details/42131581 https://blog.csdn.net/u014355480/article/details/44659245

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • 生成1~n的排列(模板),生成可重集的排列(对应紫书P184, P185)

    _DIY
  • 蓝桥杯突击复习准备——部分算法汇总

    当然,上面这个状态转移方程不适用于a数组长度较大的情况。比如AcWing896. 最长上升子序列 II (AC代码,思路在代码中)

    _DIY
  • 【Pet HDU - 4707 】【利用并查集找深度】

    _DIY
  • LeetCode 周赛 184

    判断字符串是不是子串,效率高的方式应该是字典树,按照字典序排序后,建树,再建的过程中就可以得到答案。 但是这是比赛中,又是第一题,所以直接用contains了

    ShenduCC
  • 【POJ 1062】昂贵的聘礼(最短路)

    饶文津
  • 2017.7.21夏令营清北学堂解题报告

    预计分数: 60+30+0=90=划水 实际分数: 90+30+20=140=rank5=雷蛇鼠标 一句话总结:今天该买彩票! T1: 题目描述 从前有一个?...

    attack
  • 19 nowcoder 第一场 A 笛卡尔树

    每个结点有两个键值val和key,val是它的值,key可以是任意的,这里在数组中是下标。

    用户2965768
  • BZOJ2002: [Hnoi2010]Bounce 弹飞绵羊(LCT)

    attack
  • *(int*)&p

    分析: void(p)(); 这行代码定义了一个指针变量p,p指向一个函数。这个函数的参数和返回值都为void。 &p是求指针变量p本身的地址。 ...

    Daotin
  • HDU 2011 菜鸟杯

    A:该题写了很久,一直TLE,主要是枚举到n/2时间复杂度实在太高了,其实嘛,这道题就是因式分解,所以就是i*i=n,也就是sqrt(n) #include<s...

    用户1624346

扫码关注云+社区

领取腾讯云代金券