矩阵相乘优化算法实现讲解

来源:互联网 发布:天界法宝五行进阶数据 编辑:程序博客网 时间:2024/05/22 03:04

矩阵相乘

什么是矩阵?

在数学中,矩阵(Matrix)是指纵横排列的二维数据表格,最早来自于方程组的系数及常数所构成的方阵。这一概念由19世纪英国数学家凯利首先提出。

矩阵是高等代数学中的常见工具,也常见于统计分析等应用数学学科中。并且在ACM竞赛,有很多涉及到矩阵知识的题。许多算法都会结合矩阵来处理,而比较具有代表性的矩阵算法有:矩阵快速幂、高斯消元等等。

例如下面的图片就是一个矩阵:

上述矩阵是一个 4 × 3 矩阵:

某矩阵 A 的第 i 行第 j 列,或 I , j位,通常记为 A[i,j] 或 Aij。在上述例子中 A[3,3]=2。

此外 Aij = (aij),意为 A[i,j] = aij。

 

矩阵常用知识

矩阵相乘的规则:矩阵与矩阵相乘 第一个矩阵的列数必须等于第二个矩阵的行数 假如第一个是m*n的矩阵 第二个是n*p的矩阵 则结果就是m*p的矩阵 且得出来的矩阵中元素具有以下特点:

第一行第一列元素为第一个矩阵的第一行的每个元素和第二个矩阵的第一列的每个元素乘积的和 以此类推 第i行第j列的元素就是第一个矩阵的第i行的每个元素与第二个矩阵第j列的每个元素的乘积的和。

单位矩阵: n*n的矩阵 mat ( i , i )=1; 任何一个矩阵乘以单位矩阵就是它本身 n*单位矩阵=n, 可以把单位矩阵等价为整数1。(单位矩阵用在矩阵快速幂中)

例如下图就是一个7*7的单位矩阵:

 

矩阵相乘算法实现:

① 依据简单的矩阵相乘规则我们很容易写出代码:


for(int i=0;i<n;i++)for(int j=0;j<n;j++)for(int k=0;k<n;k++)t[i][j]=(t[i][j]+x[i][k]*y[k][j]);

这个代码就是简单的计算,按照计算规则,依次算出结果矩阵的每一位元素,就是一个一个的计算出所有元素。这种思路比较简单好想,但是这种算法的复杂度是O(N3),而且不能进行优化,所以在平时在进行矩阵乘法时使用起来往往会超时。

② 我们再看一种操作方法:

一行一行的计算出所有元素:

for(int i=0;i<n;i++)for(int k=0;k<n;k++)for(int j=0;j<n;j++)t[i][j]=(t[i][j]+x[i][k]*y[k][j]);


 

试想一下:什么情况下不用再进行相乘的操作?

其实就是一个数为0的时候,这时候即使相乘也不会改变原来位置上的值。

比如,如果此时X[i][k]的值为0,那你即使进行循环操作,t[i][j]值一样不会改变。所以,当x[i][j]值为0时,我们就没必要再进行相乘操作。这样的话就会优化一定的时间。而且这种优化对于一般算法要求的时间复杂度已经足够了。

优化后代码示例:


for(int i=0;i<n;i++)for(int k=0;k<n;k++)if(x[i][k])for(int j=0;j<n;j++)t[i][j]=(t[i][j]+x[i][k]*y[k][j]);

③ 还有一种操作是一列一列的计算出所有元素:

这种情况下一样可以进行优化。


for(int j=0;j<n;j++)for(int k=0;k<n;k++)if(y[k][j])for(int i=0;i<n;i++)t[i][j]=(t[i][j]+x[i][k]*y[k][j]);

算法模板:

void Matmul(LLX[MAXN][MAXN],LL Y[MAXN][MAXN]){LL t[MAXN][MAXN]={0};for(int i=0;i<N;i++)for(int k=0;k<N;k++)if(X[i][k])for(int j=0;j<N;j++)t[i][j]=(t[i][j]+X[i][k]*Y[k][j]);for(int i=0;i<N;i++)for(int j=0;j<N;j++)X[i][j]=t[i][j];}


0 0
原创粉丝点击