【啊哈!算法】之七、动态规划---矩阵l链乘
在学习矩阵链乘之前,先要知道矩阵之间是如何进行乘法运算的,如果你对这个还不是很了解,那么先去看一下线性代数中矩阵的乘法这一节!
OK,我们来说矩阵链乘,这是动态规划算法中的一个基础问题,在算法导论15章中也详细介绍了此算法,本文也是主要参考算法导论。
我们来看一下问题: 给你N个矩阵,{A1,A2,A3,A4,......,AN},这一组矩阵必须是可以相乘的,我们要计算这N个矩阵的乘积,即:A1A2A3..AN。 因为我们的矩阵是满足结合律的,所以我们可以在计算矩阵链乘的时候可以有许多不同的计算次序。 在这里,以不同的次序做乘法,其乘法的运算次数是不同的。
来举个例子:我们来对{A1,A2,A3}做连乘,假设他们的维数分别是:10*100,100*5,5*50, 那么我们按照((A1A2)A3)的顺序来做乘法,求A1A2的乘积,也就是10*5的矩阵要做的也能算次数是10*100*5=5000次,而后再乘上A3 5*50,也就是10*5*50=2500次,总的乘法运算次数是 7500次。
我们再按照(A1(A2A3))的次序来计算,先是A2A3 的运算次数:100*5*50=25000次,而后要乘上A1 也就是 10*100*50=50000次,总的运算次数是75000次。 (⊙o⊙)… 通过上面的比较你应该会看出他们是十倍的关系!
简单一点描述矩阵链乘法的问题就是:给你n个矩阵构成的一个链{A1,A2,A3...An},其中i=1,2,3...n,矩阵Ai的维数是:pi-1 * pi,对于乘积A1A2A3...An用一种最小化标量乘法次数的方式进行加全部括号。
ok,通俗点说,我们就是要寻找最优的解,就是用最少的乘法的次数,也就是流程中乘法次数最小。
我们分三步来解决:
1、最优加全部括号的结构
动态规划的第一步就是寻找最优子结构,然后利用这个子结构就能根据子问题的最优解构造出原问题的一个最优解。
用记号A[i..j]表示乘积A[i]A[i+1]...A[j]求值的结果,其中i <= j。我们假设存在k,i <= k < j;A[i]A[i+1]...A[j]的任何全部加括号形式乘积在A[k]和A[k+1]这里分开,那么A[i]A[i+1]...A[j]的最优加全部括号的前子链A[i]A[i+1]...A[k]的加全括号必须是A[i]A[i+1]...A[k]的一个最优加全部括号。
这样一来就可以利用多得到的最优子结构来说明能够根据子问题的最优解来构造原问题的一个最优解。
2、一个递归求解
下面我们就要根据子问题的最优解来递归定义一个最优解的代价。
我们先设m[i][j]为计算矩阵A[i..j]所需的标量乘法运算次数的最小值, 对于整个问题,计算A[1..n]的最小代价就是m[1][n]。
这里,如果i==j 那么这就是一个普通问题了,矩阵链就有一个矩阵,所以不需要进行任何标量乘法来计算乘积。
但是当i<j的时候,我们必须要来计算m[i...j],我们就要利用步骤1中得到的最优解的结构了。
假设最优加全部括号将乘积A[i]A[i+1]...A[j]从A[k]和A[k+1]之间分开,i <= k < j。 因此m[i...j]就等于计算子乘积Ai...k和Ak+1...j的代价,最后还要加上一个条件是这两个矩阵相乘的代价的最小值。 我们计算Ai...kAk+1...j 要做pi-1pkpj 次标量乘法。
最后得出的公式是:

在学习矩阵链乘之前,先要知道矩阵之间是如何进行乘法运算的,如果你对这个还不是很了解,那么先去看一下线性代数中矩阵的乘法这一节!
OK,我们来说矩阵链乘,这是动态规划算法中的一个基础问题,在算法导论15章中也详细介绍了此算法,本文也是主要参考算法导论。
我们来看一下问题: 给你N个矩阵,{A1,A2,A3,A4,......,AN},这一组矩阵必须是可以相乘的,我们要计算这N个矩阵的乘积,即:A1A2A3..AN。 因为我们的矩阵是满足结合律的,所以我们可以在计算矩阵链乘的时候可以有许多不同的计算次序。 在这里,以不同的次序做乘法,其乘法的运算次数是不同的。
来举个例子:我们来对{A1,A2,A3}做连乘,假设他们的维数分别是:10*100,100*5,5*50, 那么我们按照((A1A2)A3)的顺序来做乘法,求A1A2的乘积,也就是10*5的矩阵要做的也能算次数是10*100*5=5000次,而后再乘上A3 5*50,也就是10*5*50=2500次,总的乘法运算次数是 7500次。
我们再按照(A1(A2A3))的次序来计算,先是A2A3 的运算次数:100*5*50=25000次,而后要乘上A1 也就是 10*100*50=50000次,总的运算次数是75000次。 (⊙o⊙)… 通过上面的比较你应该会看出他们是十倍的关系!
简单一点描述矩阵链乘法的问题就是:给你n个矩阵构成的一个链{A1,A2,A3...An},其中i=1,2,3...n,矩阵Ai的维数是:pi-1 * pi,对于乘积A1A2A3...An用一种最小化标量乘法次数的方式进行加全部括号。
ok,通俗点说,我们就是要寻找最优的解,就是用最少的乘法的次数,也就是流程中乘法次数最小。
我们分三步来解决:
1、最优加全部括号的结构
动态规划的第一步就是寻找最优子结构,然后利用这个子结构就能根据子问题的最优解构造出原问题的一个最优解。
用记号A[i..j]表示乘积A[i]A[i+1]...A[j]求值的结果,其中i <= j。我们假设存在k,i <= k < j;A[i]A[i+1]...A[j]的任何全部加括号形式乘积在A[k]和A[k+1]这里分开,那么A[i]A[i+1]...A[j]的最优加全部括号的前子链A[i]A[i+1]...A[k]的加全括号必须是A[i]A[i+1]...A[k]的一个最优加全部括号。
这样一来就可以利用多得到的最优子结构来说明能够根据子问题的最优解来构造原问题的一个最优解。
2、一个递归求解
下面我们就要根据子问题的最优解来递归定义一个最优解的代价。
我们先设m[i][j]为计算矩阵A[i..j]所需的标量乘法运算次数的最小值, 对于整个问题,计算A[1..n]的最小代价就是m[1][n]。
这里,如果i==j 那么这就是一个普通问题了,矩阵链就有一个矩阵,所以不需要进行任何标量乘法来计算乘积。
但是当i<j的时候,我们必须要来计算m[i...j],我们就要利用步骤1中得到的最优解的结构了。
假设最优加全部括号将乘积A[i]A[i+1]...A[j]从A[k]和A[k+1]之间分开,i <= k < j。 因此m[i...j]就等于计算子乘积Ai...k和Ak+1...j的代价,最后还要加上一个条件是这两个矩阵相乘的代价的最小值。 我们计算Ai...kAk+1...j 要做pi-1pkpj 次标量乘法。
最后得出的公式是:




2012/10/3
jofranks 于南昌