# 11 | 如何运用线性代数方法解决图论问题? 你好,我是朱维刚。欢迎你继续跟我学习线性代数,今天我要讲的内容是“如何运用线性代数方法解决图论问题”。 “图”这个字在计算机科学领域很常见,特别是在数据结构中。一说到图,是必定要联系到图论(Graph Theory)的,因为它是以图为研究对象的数学的一个分支。图论中的图,是由若干给定的**点**及连接两点的**线**所构成的图形,这种图形通常用来描述某些事物之间的某种特定关系,用点代表事物,用连接两点的线表示相应两个事物间具有这种关系。 说到这,你也许会问,这个和线性代数、矩阵有什么关系? ## 图的数学定义 既然是数学课,我们还是要先讲一下图的数学定义:一个图$G$是指一个有序三元组$(V, E, \\phi)$,$V$是非空的顶点集;$E$是不与$V$相交的边集;$\\phi$是关联函数,它使$G$的每条边对应于$G$的无序顶点对。如果$e$是一条边,$u$和$v$是顶点,使得$\\phi(e)=u v$,则$e$连接$u$和$v$,也就是顶点$u$和$v$是$e$的端点。 好了,现在是时候通过两个应用场景来看下,如何把矩阵和图论关联起来,并运用在解决实际问题中了。 ## 邻接矩阵应用 首先,是邻接矩阵(Adjacency Matrix),邻接矩阵是表示顶点之间相邻关系的矩阵。假设$G$是一个图,$V(G)$是$G$的顶点集,$E(G)$是$G$的边集,设$G$中有$n$个顶点,$v\_{1}, v\_{2}, \\ldots, v\_{n}$,$A=(a\_{ij})\_{n×n}$是$G$的邻接矩阵。 $$a\_{i}=\\left\\{\\begin{array}{l} 1, v\_{i} v\_{j} \\in E(G) \\\\\\ 0, v\_{i} v\_{j} \\notin E(G)^{\\prime}, i, j=1,2, \\ldots, n \\end{array}\\right.$$ 已知情况是这样的,那我们现在来看一下邻接矩阵在现实问题中的应用。这个例子来源于1994年全国大学生数学建模竞赛试题B题。 某厂生产一种弹子锁具,每个锁具的钥匙有5个槽,每个槽的高度从${1,2,3,4,5,6}$中任取一数。由于工艺及其他原因,制造锁具时对5个槽的高度还有两个限制。 1. 至少有3个不同的数; 2. 相邻两槽的高度之差不能为5。 满足以上条件制造出来的所有互不相同的锁具称为一批。销售部门在一批锁具中随意地取每60个装成一箱出售。问:每一批锁具有多少个,装多少箱? 我们先来看下弹子锁具的样子,否则自己想象会要一些时间。 ![](https://static001.geekbang.org/resource/image/1b/04/1b45eea656a8cc18dde70c16f5a17204.gif) 这是一个7个槽的弹子锁具,只是比我们例子的5个槽多了两个槽。有了弹子锁具形象化输出后,我们开始解题。锁具装箱是一个排列组合的数学问题,但如果我们用图论的邻接矩阵方法来解这个问题,就能够起到化繁为简的作用。 首先,我们构造一个6节点的图:把1、2、3、4、5、6这6个数作为6个节点,如果两个数字可以相邻,那这两个节点之间就加一条边,每个节点有自己到自己的一条边。于是,我们得到了锁具各槽之间的关系图: ![](https://static001.geekbang.org/resource/image/ba/0f/ba703a93fec043yyf741f221e08ca50f.png) 接着,构建邻接矩阵$A$,根据前面说的,如果两点之间有一条边,那在矩阵中,相应位置的值就是1,比如:节点1和2之间有一条边,那矩阵第一行第二列和第二行第一列的值就是1,节点1和6之间没有边,那矩阵第一行第六列和第六行第一列的值就是0,因为每个节点有自己到自己的一条边,所以第一行第一列的值就是1,其它5个节点也是一样的。 $$A=\\left\[\\begin{array}{llllll} 1 & 1 & 1 & 1 & 1 & 0 \\\\\\ 1 & 1 & 1 & 1 & 1 & 1 \\\\\\ 1 & 1 & 1 & 1 & 1 & 1 \\\\\\ 1 & 1 & 1 & 1 & 1 & 1 \\\\\\ 1 & 1 & 1 & 1 & 1 & 1 \\\\\\ 0 & 1 & 1 & 1 & 1 & 1 \\end{array}\\right\]$$ 因为我们从没有1、6相邻的关系图得到了邻接矩阵$A$,所以$A$中所有元素之和表示两个槽高**无1、6相邻**的锁具个数。而每个无1、6相邻的5位数与关系图中长度是1的一条链一一对应。于是,$A^{k}$中各元素之和就是长度为$k$的链接个数。比如,$A^{2}$中第$i$行第$j$列的元素就是$i$开始经过两条边到达$j$的链接个数。我们这里因为是5个元素,也就是要经过4条边,所以需要计算$A^{4}$。 $$A^{4}=\\left\[\\begin{array}{cccccc} 141 & 165 & 165 & 165 & 165 & 140 \\\\\\ 165 & 194 & 194 & 194 & 194 & 165 \\\\\\ 165 & 194 & 194 & 194 & 194 & 165 \\\\\\ 165 & 194 & 194 & 194 & 194 & 165 \\\\\\ 165 & 194 & 194 & 194 & 194 & 165 \\\\\\ 140 & 165 & 165 & 165 & 165 & 141 \\end{array}\\right\]$$ 把$A^{4}$中的元素求和,就能得到相邻高差为5的锁具数是6306。 最后,因为题目的限制提到了槽的高度至少有3个不同的数,我们还要把6306这个数字减去仅有一个、两个槽高的锁具:$6306-\\left(6+\\left(C\_{6}^{2}-1\\right)\\left(2^{5}-2\\right)\\right)=5880$。 所以,我们得到一批锁具的个数是5880,总共装5880/60=98箱。 这样,我们通过邻接矩阵的图论知识,解决了一批锁具的数量问题,比其它方法看起来更简单。 > 特别提示:文中用到的$A^{k}$在图论中的实际意义,是来自刘亚国的一篇文献《图论中邻接矩阵的应用》。 ## 关联矩阵应用 接下来,我们在邻接矩阵上再升级一下,把边变成有向边。这样就形成了另一类矩阵:关联矩阵。关联矩阵经常被用在图论中,现在我们就来看一下关联矩阵和图之间的关系。如下图所示,我们定义了一个拥有4个节点和6条边的图。 ![](https://static001.geekbang.org/resource/image/d4/f6/d447367b8b2980d05f44aeac421664f6.png) 接下来,定义一个6×4的矩阵来描述这个图,其中列表示点(1)到点(4),行表示边1到边6: $$A=\\left\[\\begin{array}{cccc} \-1 & 1 & 0 & 0 \\\\\\ \-1 & 0 & 1 & 0 \\\\\\ 0 & -1 & 1 & 0 \\\\\\ \-1 & 0 & 0 & 1 \\\\\\ 0 & -1 & 0 & 1 \\\\\\ 0 & 0 & -1 & 1 \\end{array}\\right\]$$ 矩阵$A$只包含了三类元素:-1、1、0。-1表示点的箭头的出方向,1表示点的箭头的入方向,0则表示点和点之间没有关联。举例来说,矩阵$A$的第一行元素是-1、1、0、0,那对于边1和点(1)、点(2)说,我们从图中可以看到边1是从点(1)到点(2),$A$中-1对于点(1)来说是箭头的出方向,1对于点(2)来说是箭头的入方向,而边1和点(3)、点(4)没有任何关系,所以第一行第三列和第四列都是0。 这里,我们把关联矩阵用到现实场景中,比如:让它为电子电路服务,用它来分析整个电路的情况,也就是电路的拓扑结构,这里的电路指的是基尔霍夫定律,是分析和计算较为复杂电路的基础。假设$x\_{1}, x\_{2}, x\_{3}, x\_{4}$是这几个点的电压值,我们来看一下$Ax$的结果: $$A x=\\left\[\\begin{array}{cccc} \-1 & 1 & 0 & 0 \\\\\\ \-1 & 0 & 1 & 0 \\\\\\ 0 & -1 & 1 & 0 \\\\\\ \-1 & 0 & 0 & 1 \\\\\\ 0 & -1 & 0 & 1 \\\\\\ 0 & 0 & -1 & 1 \\end{array}\\right\]\\left\[\\begin{array}{l} x\_{1} \\\\\\ x\_{2} \\\\\\ x\_{3} \\\\\\ x\_{4} \\end{array}\\right\]=\\left\[\\begin{array}{c} x\_{2}-x\_{1} \\\\\\ x\_{3}-x\_{1} \\\\\\ x\_{3}-x\_{2} \\\\\\ x\_{4}-x\_{1} \\\\\\ x\_{4}-x\_{2} \\\\\\ x\_{4}-x\_{3} \\end{array}\\right\]$$ 由此可见,结果是差值,也就是沿着边1到6的电势差,有了电势差,就说明有电流,但如果Ax=0会怎样呢?也就是方程满足这样的等式: $$A x=\\left\[\\begin{array}{cccc} \-1 & 1 & 0 & 0 \\\\\\ \-1 & 0 & 1 & 0 \\\\\\ 0 & -1 & 1 & 0 \\\\\\ \-1 & 0 & 0 & 1 \\\\\\ 0 & -1 & 0 & 1 \\\\\\ 0 & 0 & -1 & 1 \\end{array}\\right\]\\left\[\\begin{array}{l} x\_{1} \\\\\\ x\_{2} \\\\\\ x\_{3} \\\\\\ x\_{4} \\end{array}\\right\]=\\left\[\\begin{array}{c} x\_{2}-x\_{1} \\\\\\ x\_{3}-x\_{1} \\\\\\ x\_{3}-x\_{2} \\\\\\ x\_{4}-x\_{1} \\\\\\ x\_{4}-x\_{2} \\\\\\ x\_{4}-x\_{3} \\end{array}\\right\]=\\left\[\\begin{array}{l} 0 \\\\\\ 0 \\\\\\ 0 \\\\\\ 0 \\\\\\ 0 \\\\\\ 0 \\end{array}\\right\]$$ 很明显,这些差值,也就是电势差都等于0,意味着没有电流。同理,如果把电压值换成温度呢?那应用场景就切换成热传导了。 刚才我们看到了$Ax=0$的情况,你还记得[第八篇](https://time.geekbang.org/column/article/272815)中说的零空间吗?它关注的就是$Ax=0$,也就是向量空间$V$中所有经过$\\phi$映射为零的向量集合,用符号表示就是:$ker(\\phi)$,它的维数叫做零化度(nullity),表示成:$dim(ker(\\phi))$。 而在电路例子中,它表示的是所有六个电势差都是0,也就意味着:所有四个电压值是相等的,在零空间中的每个$x$都是一个常向量:$x=(c,c,c,c)$。所以,$A$的零空间是一条线。无论我们怎么同时升高或降低电压量$c$,都不会改变电势差0。 刚才我们说的是电压,现在我们来具体看看关联矩阵在基尔霍夫电流定律上的运用。我们来定义一个拥有4个节点和5条边的图: ![](https://static001.geekbang.org/resource/image/ce/d6/ce37e3d70213cfc23f6c33eb00637ed6.png) 基尔霍夫电流定律定义:$A^{T} y=0$,其中y是向量$y\_{1}, y\_{2}, y\_{3}, y\_{4}, y\_{5}$,我们把这个图以关联矩阵的形式写出来就是: $$\\left\[\\begin{array}{ccccc} \-1 & 0 & -1 & -1 & 0 \\\\\\ 1 & -1 & 0 & 0 & 0 \\\\\\ 0 & 1 & 1 & 0 & -1 \\\\\\ 0 & 0 & 0 & 1 & 1 \\end{array}\\right\]\\left\[\\begin{array}{l} y\_{1} \\\\\\ y\_{2} \\\\\\ y\_{3} \\\\\\ y\_{4} \\\\\\ y\_{5} \\end{array}\\right\]=\\left\[\\begin{array}{c} 0 \\\\\\ 0 \\\\\\ 0 \\\\\\ 0 \\end{array}\\right\]$$ 这里-1,0,1的含义上面有所描述,第一行和$y$向量相乘后得到:$-y\_{1}-y\_{3}-y\_{4}=0$,说明从节点1出来的总电流等于0,满足守恒定律;第二行和$y$向量相乘后得到:$y\_{1}-y\_{2}=0$,说明流入节点2的电流和从节点2流出的电流相等;同样,后面两行分别和$y$向量相乘后得到:$y\_{2}+y\_{3}-y\_{5}=0$和$y\_{4}+y\_{5}=0$,和图表示的都一致,也都符合守恒定律。 好了,到这里简单电路的数学知识,也就是关联矩阵讲完了,如果碰到更复杂的电路,比如:在节点之间,也就是边上有电流源,那么,等式就要从$A^{T} y=0$变成$A^{T} y=f$。 ## 本节小结 本节是第一篇应用篇,所以我从更贴近生活的例子来讲解线性代数的应用,通过弹子锁具,让你能够了解,邻接矩阵与图论之间是怎么关联的;通过基尔霍夫定律,让你能够了解关联矩阵与图论之间是怎么关联的。 所以,邻接矩阵、关联矩阵的最大意义就是用数学的方式描述图,进而来描述某些事物之间的某种特定关系,是不是发现问题后通过数学工具来解决问题很美妙呢? ## 线性代数练习场 这次的练习题稍微有些难度,是一道传统练习题。 三名商人各带一个随从乘船渡河,现有一只小船只能容纳两个人,由他们自己划行,若在河的任一岸的随从人数多于商人,他们就可能抢劫财物。但如何乘船渡河由商人决定,试给出一个商人安全渡河的方案,使得渡河的次数最少。 > 注意:这里的问题包含两层含义——安全渡河和渡河次数最少。 > 提示:使用本节的第一个例子的邻接矩阵和$A^{k}$来解这道题。 欢迎在留言区晒出你的运算过程和结果,我会及时回复。同时,也欢迎你把这篇文章分享给你的朋友,一起讨论、学习。