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;
}


原文地址:https://www.cnblogs.com/riskyer/p/3246565.html