HDOJ5015 233 Matrix(矩阵乘法加速递推)

原题链接:HDOJ5015

题意简述:给定矩阵每个元素计算公式,求下标为(m,m)的元素的值。

解题思路:刚开始想着递推求出整个矩阵,后来我哭了。m 可达1e9,要化简到 lg 级别才能不超时,这时我隐约想到用转移矩阵加快速幂做了,根据《算法竞赛进阶指南》P151页所述,如果一类问题具有如下特点:

  • 可以抽象出一个长度为n的一维向量,该向量在每个单位时间发生一次变化
  • 变化的形式是一个线性递推(只有若干“加法”或者“乘一个系数”的运算)
  • 该递推式在每个时间可能作用于不同的数据上,但本身保持不变
  • 向量变化时间(即递推的轮数)很长,但向量长度n不大。

那么就可以考虑采用矩阵乘法进行优化。具体步骤就是构造出转移矩阵、状态矩阵,通过对转移矩阵的快速幂来加速递推。时间复杂度为(n^3 lgT),其中T为递推轮数。

代码示例(如何构造转移矩阵于状态矩阵就不写了,手太冷了,而且我也讲不好):

#include<cstdio>
#include<cstring>
const int maxn = 50;
const int M = 10000007;
struct Matrix{
	int n;
	int v[maxn][maxn];
	Matrix(int n){
		this->n = n;
		memset(v,0,sizeof v);
	}
	Matrix operator *(const Matrix &b) const{
		Matrix C(n);
		for(int i = 0;i < n;i++)
		for(int j = 0;j < n;j++)
			for(int k = 0;k < n;k++)
				C.v[i][j] = (C.v[i][j] + 1ll*v[i][k]*b.v[k][j]%M) % M;
		return C;
	}
	Matrix operator +(const Matrix &b) const{
		Matrix C(n);
		for(int i = 0;i < n;i++)
		for(int j = 0;j < n;j++)
			C.v[i][j] = (v[i][j] + b.v[i][j])%M;
		return C;
	}
	void qpow(int k){
		Matrix E(n);
		Matrix A = *this;
		for(int i = 0;i < n;i++) E.v[i][i] = 1;
		while(k){
			if(k&1) E = E*A;
			A = A*A;
			k >>= 1;
		}
		*this = E;
	}
};
int main(){
	int n,m;
	while(scanf("%d%d",&n,&m) != EOF){
		Matrix tran(n+2);
		for(int i = 0;i <= n;i++){
			tran.v[i][0] = 10;
			tran.v[i][n+1] = 1;
		}
		for(int i = 1;i <= n;i++)
			for(int j = 1;j <= i;j++)
				tran.v[i][j] = 1;
		tran.v[n+1][n+1] = 1;
		tran.qpow(m);
		int t[n+2],ans[n+2];
		memset(ans,0,sizeof ans);
		t[0] = 23,t[n+1] = 3;
		for(int i = 1;i <= n;i++)	scanf("%d",&t[i]);
		for(int i = 0;i < n+2;i++)
			for(int j = 0;j < n+2;j++)
				ans[i] = (ans[i] + 1ll*tran.v[i][j]*t[j]%M)%M;
		printf("%d\n",ans[n]);
	}
	
}

 

posted @ 2019-01-16 15:12  Dr_Lo  阅读(144)  评论(0编辑  收藏  举报