[洛谷P4111][HEOI2015]小Z的房间

题目大意:有一个$n\times m$的房间,一些位置是房间,另一些位置是柱子,相邻两个房间之间有墙,问有多少种方案可以打通一些墙把所有房间连成一棵树,柱子不可以打通

题解:矩阵树定理,把房间当点,墙当边,一张图的生成树个数为每个点的度数矩阵减去邻接矩阵的任意一个代数余子式的值。

模数是$10^9$,不可以直接高斯消元,可以用辗转相除法来消元

卡点:

 

C++ Code:

#include <algorithm>
#include <cstdio>
#include <cstring>
#define maxn 10
const int mod = 1e9;

int n, m;
int s[maxn][maxn], idx;
char buf[maxn];

inline void up(int &a, int b) {a += b - mod, a += a >> 31 & mod;}
inline void down(int &a, int b) {a -= b, a += a >> 31 & mod;}
struct Deter {
	#define maxm 85
	int s[maxm][maxm];
	inline void addedge(int x, int y) {
		s[x][x]++, s[y][y]++;
		s[x][y] = s[y][x] = mod - 1;
	}
	int gauss() {
		int f = 0, ans = 1;
		for (int i = 1; i < idx; i++) {
			for (int j = i + 1; j < idx; j++) {
				while (s[j][i]) {
					int tmp = s[i][i] / s[j][i];
					for (int k = i; k < idx; k++) {
						down(s[i][k], static_cast<long long> (s[j][k]) * tmp % mod);
						std::swap(s[i][k], s[j][k]);
					}
					f ^= 1;
				}
			}
			ans = static_cast<long long> (ans) * s[i][i] % mod;
			if (!ans) return 0;
		}
		return f ? mod - ans : ans;
	}
	#undef maxm
} D;

int main() {
	memset(s, -1, sizeof s);
	scanf("%d%d", &n, &m);
	for (int i = 0; i < n; i++) {
		scanf("%s", buf);
		for (int j = 0; j < m; j++) if (buf[j] == '.') {
			s[i][j] = idx++;
			if (i && ~s[i - 1][j]) D.addedge(s[i - 1][j], s[i][j]);
			if (j && ~s[i][j - 1]) D.addedge(s[i][j - 1], s[i][j]);
		}
	}
	printf("%d\n", D.gauss());
	return 0;
}

  

posted @ 2018-12-03 19:26  Memory_of_winter  阅读(156)  评论(0编辑  收藏  举报