zoukankan      html  css  js  c++  java
  • 矩阵快速幂

    一、基础知识

    1、矩阵的定义

    !矩阵定义

    2、行矩阵和列矩阵应该比较好理解,就是一个矩阵的每一行都可以称之为一个行矩阵,同理列矩阵也一样。

    3、同型矩阵:设有矩阵A和矩阵B,矩阵A的行数和列数都与矩阵B的相同,则矩阵A、B是同型矩阵

    像图中这样就是两个同型矩阵

    同型矩阵

    4、单位矩阵

    在矩阵的乘法中,有一种矩阵起着特殊的作用,如同数的乘法中的1,这种矩阵被称为单位矩阵.

    它是个方阵,从左上角到右下角的对角线(称为主对角线)上的元素均为1。除此以外全都为0。

    二、矩阵的相关运算

    1、矩阵加法

    矩阵加法

    图上说的也很清楚了吧,矩阵加法满足加法交换律和加法结合律。

    2、矩阵乘法

    结合图片理解一下吧。
    矩阵乘法

    其中,c[i][j]表示:矩阵A的第 i 行与矩阵B的第 j 列的对应乘积的和。

    也就是c数组

    2、矩阵快速幂引入

    为了引出矩阵快速幂,我们先来学整数快速幂,加入我们要计算如果现在要算X^8:则 XXXXXXXX 按照寻常思路,一个一个往上面乘,则乘法运算进行7次。

    (XX)(XX)(XX)(XX)
    这种求法,先进行乘法得X2,然后对X2再执行三次乘法,这样去计算,则乘法运算执行4次。已经比七次要少。所以为了快速算的整数幂,就会考虑这种结合的思想。

    现在要考虑应该怎么分让计算比较快。接下来计算整数快速幂。例如:X^19次方。

    19的二进制为:1 0 0 1 1 。

    由(Xm)(Xn) = X^(m+n)

    则X^19 = (X16)(X2)*(X^1)

    那么怎么来求解快速幂呢。请看下列代码:
    求解X^N的值。

    int quickpow(int x,int N){
    	int res = x;
        int ans = 1;
        while(N){
        	if(N & 1){
            	ans = ans * res;
            }
            res = res * res;
            N = N >> 1;
        }
        return ans;
    }
    

    那现在,我们来仔细看一下这个代码。

    对于x^19来说,19的二进制是1 0 0 1 1

    初始化

    ans = 1;  res = x;
    

    因为19的二进制最后一位是 1 ,所以19是奇数。

    ans = res * ans = x ; 
    
    res = res * res = x ^ 2;
    

    然后右移一位,就变成了1 0 0 1

    则最后一位是 1 ,所以是奇数。

    ans = ans * res 
        = x * (x ^ 2) = x ^ 3;
    
    res = res * res
        = x ^ 2 * x ^ 2
        = x ^ 4;
    

    然后右移一位,变成了1 0 0

    则最后一位是 0 ,所以是偶数。

    res = res * res 
        = x ^ 4 * x ^ 4
        = x ^ 8;
    

    然后右移一位,变成了1 0

    则最后一位是偶数。

    res = res * res 
        = x ^ 8 * x ^ 8
        = x ^ 16;
    
    

    然后右移一位,只剩下 1 了。

    还看吗?肯定是奇数

    ans = ans * res 
        = (x ^ 3) *(x ^ 16) 
        = x ^ 19
        
    res = res * res = x ^ 32
    

    总结一下:

      从上述叙述中可以看出res = x ^ m
      始终是与二进制位置上的权值是相对应的。
      
      当二进制位是 0 时 , 我们只让res ^ 2,来对应二进制下一位的权值
      当二进制位是 1 时 , ans = ans * res , 再使res ^ 2
    

    说了一大堆,我们进入正题,矩阵快速幂

    假如现在有一个n * n的方阵A。给出一个数M,让算矩阵A的M次幂,A ^ M.则上面代码可以化为。

    struct matrix{
    	int m[maxn][maxn];
    }ans , res;//这是计算矩阵乘法的参数
    
    //我们假设参数矩阵是 A 、B和一个数n(表示n * n)
    
    matrix mul(matrix A,matrix B){
    	matrix tmp;
        //这是我们临时设置的一个矩阵,存A*B的结果
        for(int i=1;i<=n;i++)
          for(int j=1;j<=n;j++)
            tmp.m[i][j] = 0;
        for(int i=1;i<=n;i++)
          for(int j=1;j<=n;j++)
            for(int k=1;k<=n;k++)
              tmp.m[i][j] += A.m[i][k] * B.m[k][j];
        return tmp;
    }
    
    //快速幂算法,求矩阵res的N次幂
    
    matrix quickpower(matrix res,int x){
    	/*我们在上面介绍整数幂的时候将ans初始化为 1
          但对于矩阵乘法来说,ans应该初始化为单位矩阵
          现在补充一个单位矩阵的性质:
          单位矩阵乘任意矩阵都是原矩阵
        */
        for(int i=1;i<=n;i++)//单位矩阵定义方法
          for(int j=1;j<=n;j++){
            if(i == j)  ans.m[i][j] = 1;
            else  ans.m[i][j] = 0;
          }
        while(x){
          if(x & 1)
             ans = mul(ans , res);
          res = mul(res , res);
          x = x >> 1;
        }
    }
    

    贴上一个完整的代码吧:
    P3390 矩阵快速幂模板

    #include <cstdio>
    #include <istream>
    #include <cstring>
    #include <algorithm>
    #include <cmath>//pow函数,其实没啥用 
    using namespace std;
    const int mod = pow(10,9)+7;
    
    long long n,k;
    struct matrix{
    	long long m[105][105];
    }ans , res;
    
    matrix mul(matrix x,matrix y){
        matrix tmp;
    	for(int i=1;i<=n;i++)
          for(int j=1;j<=n;j++)
            tmp.m[i][j] = 0;
        for(int i=1;i<=n;i++)
          for(int j=1;j<=n;j++)
            for(int k=1;k<=n;k++)
              tmp.m[i][j] = tmp.m[i][j] % mod + (x.m[i][k] * y.m[k][j]) % mod;
        return tmp;
    }
    
    matrix quickpower(matrix res,long long k){
        for(int i=1;i<=n;i++)
          for(int j=1;j<=n;j++){
            if(i == j)  ans.m[i][j] = 1;
            else  ans.m[i][j] = 0;
          }
        while(k){
            if(k & 1)
              ans = mul(ans , res);
            res = mul(res , res);
            k = k >> 1;
        }
        return ans;
    }
    
    int main(){
        scanf("%lld%lld",&n,&k);
        for(int i=1;i<=n;i++){ 
            for(int j=1;j<=n;j++)
                scanf("%lld",&res.m[i][j]);
        }       
        ans = quickpower(res , k);
        for(int i=1;i<=n;i++){
            for(int j=1;j<=n;j++)
                printf("%lld ",ans.m[i][j] % mod);
            printf("
    ");
        }
        return 0;   
    }
    

    当然,矩阵快速幂还有另一种写法,用重载运算符重载 * 嘛

    代码我就不解释了

    // luogu-judger-enable-o2
    #include<iostream>
    #include<cstdio>
    #include<cstring>
    #include<algorithm>
    
    #define ll long long
    
    using namespace std;
    const int N = 110;
    const int MOD = 1e9 + 7;
    
    ll n,k;
    struct martix {
        ll a[N][N];
        martix operator*(const martix &b) const {
            martix x;
            for(int i = 1 ; i <= n ; i++) {
                for(int j = 1 ; j <= n ; j++) {
                    x.a[i][j] = 0;
                    for(int k = 1 ; k <= n ; k++) {
                        x.a[i][j] += a[i][k] * b.a[k][j];
                        x.a[i][j] %= MOD;
                    }
                }
            }   
            return x;
        }
    }mart;
    martix quick_pow(martix x,ll y) {
        martix ans = x,s = x;
        for( ; y ; y >>= 1) {
            if(y & 1) ans=ans * s;
            s = s * s;
        }
        return ans;
    }
    
    int main() {
        scanf("%lld%lld",&n,&k);
        for(int i = 1 ; i <= n ; i++) {
            for(int j = 1 ; j <= n ; j ++) {
                scanf("%lld",&mart.a[i][j]);
            }
        }
        mart = quick_pow(mart,k - 1);
        for(int i = 1 ; i <= n ; i++) {
            for(int j = 1 ; j <= n ; j ++) 
                printf("%lld ",mart.a[i][j]);
            printf("
    ");
        }
        return 0;
    }
    

    上面只是简单的计算矩阵的幂,可能会感觉很抽象,因为上述矩阵并没有具体的含义,
    现在就举例说明矩阵快速幂在实际运用中的意义:

    以最常见的斐波那契数列为例:众所周知:斐波那契数列递推公式为:
    F[n] = F[n-1] + F[n-2].

    由f[0]=0,f[1]=1,可以递推后面的所有数。

    在以前,我们会常常用for循环,这是最直接的算法。

    但是,n 等于十亿你该怎么用for循环呢。。

    好吧,在介绍矩阵快速幂,肯定是用矩阵快速幂来写。。

    顺风不浪,逆风不怂。
  • 相关阅读:
    异步请求模板和数据
    关于线上js报错问题的思考
    标准web浏览器的组件
    监控图片加载的方法
    图片轮播
    jQuery的一些小技巧()
    setTimeout/setInterval伪异步
    linux/windows java jdk环境配置
    iOS pod 第三方 unrecognized selector sent to instance
    上海4
  • 原文地址:https://www.cnblogs.com/Stephen-F/p/9883245.html
Copyright © 2011-2022 走看看