• poj3233之经典矩阵乘法


    Matrix Power Series
    Time Limit: 3000MS   Memory Limit: 131072K
    Total Submissions: 12346   Accepted: 5262

    Description

    Given a n × n matrix A and a positive integer k, find the sum S = A + A2 + A3 + … + Ak.

    Input

    The input contains exactly one test case. The first line of input contains three positive integers n (n ≤ 30), k (k ≤ 109) and m (m < 104). Then follow n lines each containing n nonnegative integers below 32,768, giving A’s elements in row-major order.

    Output

    Output the elements of S modulo m in the same way as A is given.

    Sample Input

    2 2 4
    0 1
    1 1

    Sample Output

    1 2
    2 3

    我比较喜欢用数组做矩阵乘法而不是用结构体做,而且用数组做效率更高,本代码poj跑了460ms,代码中有详细注释

    #include<iostream>
    #include<cstdio>
    #include<cstdlib>
    #include<cstring>
    #include<string>
    #include<queue>
    #include<algorithm>
    #include<map>
    #include<iomanip>
    #define INF 99999999
    using namespace std;
    
    const int MAX=30+10;
    __int64 array[MAX][MAX],sum[MAX][MAX],temp[MAX][MAX],ans[MAX][MAX];
    int n,m,k;
    //array记录初始矩阵,sum记录每次幂乘后的矩阵,temp临时矩阵,ans记录结果矩阵 
    
    void MatrixInit(__int64 a[MAX][MAX],bool flag){//初始化矩阵,可以初始化为1或者A 
    	for(int i=0;i<n;++i){
    		for(int j=0;j<n;++j){
    			if(flag)a[i][j]=array[i][j];//初始化为矩阵A(A是指题目中的A,也就是这里的array) 
    			else a[i][j]=(i == j);//初始化为矩阵1 
    		}
    	}
    }
    
    void MatrixAdd(__int64 a[MAX][MAX],__int64 b[MAX][MAX],int &mod){//矩阵相加,a为相加后的矩阵 
    	for(int i=0;i<n;++i){
    		for(int j=0;j<n;++j){
    			a[i][j]=(a[i][j]+b[i][j])%mod;
    		}
    	}
    }
    
    void MatrixMult(__int64 a[MAX][MAX],__int64 b[MAX][MAX],int &mod){//矩阵相乘,a为相乘后的矩阵 
    	__int64 c[MAX][MAX]={0};
    	for(int i=0;i<n;++i){
    		for(int j=0;j<n;++j){
    			for(int k=0;k<n;++k){
    				c[i][j]+=a[i][k]*b[k][j];
    			}
    		}
    	}
    	for(int i=0;i<n;++i){
    		for(int j=0;j<n;++j)a[i][j]=c[i][j]%mod;
    	}
    }
    
    void MatrixPow(int k,int &mod){//矩阵快速幂 
    	MatrixInit(sum,0);//初始化sum为1
    	MatrixInit(temp,1);//初始化temp为A 
    	while(k){
    		if(k&1)MatrixMult(sum,temp,mod);
    		MatrixMult(temp,temp,mod);
    		k>>=1;
    	}
    }
    
    void MatrixSum(int k,int &mod){//矩阵和 
    	if(k == 1){MatrixInit(ans,1);return;}
    	MatrixSum(k/2,mod);
    	MatrixPow((k+1)/2,mod);
    	if(k&1){//k为奇数则A+(A+A^m)*(A+A^2+A^3...),m=(k+1)/2
    		MatrixInit(temp,1);//初始化temp为A
    		MatrixAdd(sum,temp,mod);//计算A+A^m,m=(k+1)/2
    		MatrixMult(ans,sum,mod);//计算(A+A^m)*(A^1+A^2+...)
    		MatrixAdd(ans,temp,mod);//计算A+(A+A^m)*(A^1+A^2...) 
    	} 
    	else{//k为偶数则(1+A^m)*(A+A^2+A^3...),m=(k+1)/2
    		MatrixInit(temp,0);//初始化temp为1 
    		MatrixAdd(temp,sum,mod);//计算1+A^m,m=(k+1)/2
    		MatrixMult(ans,temp,mod);//计算(1+A^m)*(A^1+A^2+...) 
    	}
    }
    
    int main(){
    	scanf("%d%d%d",&n,&k,&m);
    	for(int i=0;i<n;++i){
    		for(int j=0;j<n;++j){
    			scanf("%I64d",&array[i][j]);
    			ans[i][j]=0;//ans是矩阵和 
    		}
    	}
    	MatrixSum(k,m);//矩阵和:A^1+A^2+A^3+A^4+A^5+A^6=(A^1+A^2+A^3)+A^3(A^1+A^2+A^3)
    	for(int i=0;i<n;++i){
    		for(int j=0;j<n-1;++j)printf("%d ",ans[i][j]);
    		printf("%I64d
    ",ans[i][n-1]);
    	} 
    	return 0;
    }


  • 相关阅读:
    【BZOJ1345】[Baltic2007] 序列问题(单调栈大水题)
    【BZOJ2940】[POI2000] 条纹(Multi-SG)
    【BZOJ4589】Hard Nim(FWT+快速幂)
    【CF438E】The Child and Binary Tree(生成函数+多项式开根)
    【洛谷5205】【模板】多项式开根
    【BZOJ4036】[HAOI2015] 按位或(Min-Max容斥+FWT)
    【BZOJ4381】[POI2015] ODW(设阈值+倍增)
    【BZOJ3328】PYXFIB(矩乘+单位根反演)
    【BZOJ2674】Attack(整体二分+树状数组套线段树)
    单纯看懂公式的单位根反演
  • 原文地址:https://www.cnblogs.com/riskyer/p/3246565.html
Copyright © 2020-2023  润新知