AcWing 372. 棋盘覆盖 二分图最大匹配

本文详细解析了AcWing372题目的解决方案——棋盘覆盖问题,通过构建二分图并利用匈牙利算法求解最大匹配问题,实现了对有缺陷的棋盘的有效覆盖。

AcWing 372. 棋盘覆盖 二分图最大匹配

372. 棋盘覆盖

Solution

  • 将每个格子看成一个点,占据两个格子的小方块看成一条边,这个问题就可以看成最多可以取多少条边,使得所有取出的边没有公共点,就等价于做一个最大匹配问题

那么,现在的问题是,这个图是不是二分图

  • 这是一个经典做法,对这个 n × m n\times m n×m 的矩形进行二染色,发现这个二染色是没有矛盾的,所以这个图就是一个二分图,因此我们可以用匈牙利算法求最大匹配。
image-20210310162950823

现在,把 i + j i+j i+j 为偶数的点放在一边,为奇数的点放在另外一边,在枚举一边的点求最大匹配即可。

Code

#include <bits/stdc++.h>
using namespace std;
typedef long long LL;
typedef unsigned long long ULL;
typedef pair<int, int> PII;
#define debug(a) cout << #a << " " << a << endl
const int maxn = 150;
const int N = 150, M = N * 2;
const int inf = 0x3f3f3f3f;
const long long mod = 1e9 + 7;
inline long long read();

int n, m;
PII match[N][N];
bool g[N][N], st[N][N];
//st 是存储这个点是否在匈牙利算法中被找过
//g 是存储这个点是否是坏点

int dx[4] = { -1, 0, 1, 0}, dy[4] = {0, 1, 0, -1};

bool find(int x, int y) {
	for(int i = 0; i < 4; i++) {//在满足条件的方向上取点
		int a = x + dx[i], b = y + dy[i];
		if(a < 1 || a > n || b < 1 || b > n) continue;
		if(st[a][b] || g[a][b]) continue;
		st[a][b] = true; //说明(x,y)可以成功配对(a,b),标记(a,b) 现在已经名花有主
		PII t = match[a][b];
		if(t.first == -1 || find(t.first, t.second)) {//如果这个点还没有被分配或者已经有人和(a,b) 配对,但是和(a,b)配对的那个人(即t)还有其他选择
			match[a][b] = {x,y}; //t放弃了(a,b),把(a,b)让给当前追求(a,b)的人
			return true;
		}
	}

	return false;
}


int main() {

//	freopen("input.txt", "r", stdin);
//	freopen("output.txt", "w", stdout);

//	ios::sync_with_stdio(false);
	cin >> n >> m;

	while(m--) {
		int x, y;
		cin >> x >> y;
		g[x][y] = true;
	}

	memset(match, -1, sizeof match);//初始化匹配数组,-1表示没有被匹配过

	int ans = 0;
	for(int i = 1; i <= n; i++) {
		for(int j = 1; j <= n; j++) {
			if((i + j) % 2 != 0 && !g[i][j]) {
				memset(st, 0, sizeof st);//初始化st 数组表示这一轮中有没有被找过
				if(find(i, j)) ans++;
			}
		}
	}

	cout << ans << '\n';





	return 0;
}


/*
数组开够了吗 开到上界的n+1次方
初始化了吗
*/







inline LL read() {
	char ch = getchar();
	LL p = 1, data = 0;
	while(ch < '0' || ch > '9') {
		if(ch == '-')p = -1;
		ch = getchar();
	}
	while(ch >= '0' && ch <= '9') {
		data = data * 10 + (ch ^ 48);
		ch = getchar();
	}
	return p * data;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值