矩阵树定理(+行列式)

command_block

2019-12-11 13:00:14

题解

本文同时也作为P6178的题解。

1.行列式基础

对于一个矩阵 A[1...n][1...n] ,其行列式为

\det(A)=\sum\limits_{P}(-1)^{\mu(P)}\prod\limits_{i=1}^nA[i][p_i]

(枚举排列 P[1...n] ,其中 \mu(P) 为排列 P 的逆序对数)

( \det(A) 又作 |A| )

也就是说在每一挑一个乘起来,然后再拿逆序对有关的东西做系数。

直接按照定义计算,复杂度是 O(n!·n) 的。

知道了下面几条性质之后,我们就能快速计算行列式了。

这个比较显然,因为 P 除了对角线之外没有别的选择,而对角线乘积为 1

类似的,上三角矩阵和下三角矩阵的行列式都是对角线乘积。

(根据逆序对感性理解即可)

我们考虑对于每个排列,这一次交换只会影响乘积组前面的逆序对系数。

对于任意一个序列,交换两个数对于逆序对个数的影响必定为奇数(可以自己讨论一下)。

所以逆序对系数全部都改变,即变号。

这个比较好理解,因为这一行选且只选一个嘛。

\begin{vmatrix}a+a'&b+b'\\c&d\end{vmatrix}=\begin{vmatrix}a&b\\c&d\end{vmatrix}+\begin{vmatrix}a'&b'\\c&d\end{vmatrix}

还是因为一行选且只选一个,然后乘法分配律就好了。

(2,3合起来就是的线性性)

这个证明比较巧妙 : 考虑交换相同的这两行,根据(1)得行列式变号

但是矩阵并没有实质的改变,行列式不变,所以行列式的值只能是 0.

考虑构造(省略的部分表示不变):

欲证\begin{vmatrix}a&b&c\\.&.&.\\d&e&f\end{vmatrix}=\begin{vmatrix}a&b&c\\.&.&.\\d+ka&e+kb&f+kc\end{vmatrix}

首先构造一个\begin{vmatrix}a&b&c\\.&.&.\\a&b&c\end{vmatrix},根据(4)得其值为0.

又根据(2)得\begin{vmatrix}a&b&c\\.&.&.\\ka&kb&kc\end{vmatrix}也为0。

然后根据(3)得

\begin{vmatrix}a&b&c\\.&.&.\\d+ka&e+kb&f+kc\end{vmatrix}=\begin{vmatrix}a&b&c\\.&.&.\\d&e&f\end{vmatrix}+\begin{vmatrix}a&b&c\\.&.&.\\ka&kb&kc\end{vmatrix}=\begin{vmatrix}a&b&c\\.&.&.\\d&e&f\end{vmatrix}

其实我们上面证明的性质,就是为这个来打基础的。

我们消元的时候,只是用了 {交换两行, 某一行乘一个常数, 一行加上另一行的倍数} 三种操作。

那么,我们对于 A 进行消元,然后记录操作对于行列式的影响,最后得到上三角矩阵,行列式就是对角线乘积,如果消不出来那么返回 0

最后再把影响逆回去即可。

2.矩阵树定理与部分扩展

(默认图中无自环)

给出一个无向无权图,设 A 为邻接矩阵, D 为度数矩阵( D[i][i]=节点 i 的度数,其他的无值)。

则基尔霍夫(Kirchhoff)矩阵即为 : K=D-A

然后令 K'K 去掉第k行与第k列(k任意)的结果(n-1阶主子式),

~~证明不会。感兴趣的同学可以自行百度。~~ **例题** : [P4336 [SHOI2016]黑暗前的幻想乡](https://www.luogu.org/problemnew/show/P4336) - ## 加权扩展 容易理解 : 带重边的情况,上面的经典矩阵树定理也是能够处理的。 根据乘法原理,对于某种生成树的形态,其贡献为**每条边重的次数的乘积**。 如果把重边次数理解成权值的话,那么矩阵树定理求的就是 : 所有生成树边权乘积的总和。 (这里注意度数矩阵变成了相邻边的权值和) **例题** : [P3317 [SDOI2014]重建](https://www.luogu.com.cn/problem/P3317) - ## 有向扩展 前面都是无向图,神奇的是有向图的情况也是可以做的。 (邻接矩阵 $A$ 的意义同有向图邻接矩阵) 那么现在的矩阵 $D$ 就要变一下了。 若 $D[i][i]=\sum\limits_{j=1}^nA[j][i]$ ,即**到该点的边权总和(入)**。 此时求的就是**外向树** (从根向外) 若 $D[i][i]=\sum\limits_{j=1}^nA[i][j]$ ,即从**从该点出发的边权总和(出)**。 此时求的就是**内向树** (从外向根) (如果考场上不小心忘掉了,可以手玩小样例) (同样可以加权!) 此外,既然是有向的,那么就需要**指定根**。 前面提过要任意去掉第 $k$ 行与第 $k$ 列,是因为无向图所以不用在意谁为根。 在有向树的时候需要理解为指定根,结论是 : 去掉哪一行就是那一个元素为根。 **例题** : [P4455 [CQOI2018]社交网络](https://www.luogu.com.cn/problem/P4455) (内向树) 本题就是外向树,下面给出代码(出乎意料,跑的还挺快)。 ```cpp #include<algorithm> #include<cstdio> #define ll long long #define mod 1000000007 #define Maxn 305 using namespace std; ll powM(ll a,int t=mod-2) { ll ans=1; while(t){ if (t&1)ans=ans*a%mod; a=a*a%mod;t>>=1; }return ans; } int n,m; ll *a[Maxn],_a[Maxn][Maxn]; ll det(ll **a) { ll ans=1;bool tr=0; for (int j=1;j<n;j++){ for (int i=j;i<n;i++) if (a[i][j]){ if (i!=j){ swap(a[i],a[j]); tr=!tr; }break; } if (a[j][j]==0)return 0; ans=ans*a[j][j]%mod; ll t=powM(a[j][j]); for (int k=j;k<n;k++)a[j][k]=t*a[j][k]%mod; for (int i=j+1;i<n;i++){ t=mod-a[i][j]; for (int k=j;k<n;k++) a[i][k]=(a[i][k]+t*a[j][k])%mod; } }return tr? (mod-ans)%mod : ans; } int op; int main() { scanf("%d%d%d",&n,&m,&op); for (int i=1;i<=n;i++)a[i]=_a[i]; for (int i=1,f,t,w;i<=m;i++){ scanf("%d%d%d",&f,&t,&w); if (f==n)f=1;else if (f==1)f=n; if (t==n)t=1;else if (t==1)t=n; if (op==1){ a[f][t]=(a[f][t]-w+mod)%mod; a[t][t]=(a[t][t]+w)%mod; }else { a[f][t]=(a[f][t]-w+mod)%mod; a[t][t]=(a[t][t]+w)%mod; a[t][f]=(a[t][f]-w+mod)%mod; a[f][f]=(a[f][f]+w)%mod; } }printf("%lld",det(a)); return 0; } ```