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

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;
}
堆排序是一种高效的排序算法,它利用了堆的数据结构来实现排序。堆是一个完全二叉树,具有以下性质:对于任意节点 i,其父节点的值小于等于子节点的值。 堆排序的基本思路如下: 1. 构建最大堆:将待排序的数组看作是一个完全二叉树,从最后一个非叶子节点开始,依次向上调整每个节点,使得以该节点为根的子树满足最大堆的性质。 2. 将堆顶元素与最后一个元素交换:将最大堆的堆顶元素(即数组的第一个元素)与数组最后一个元素交换位置,此时最大元素就位于数组的最后。 3. 调整堆:将剩余元素重新调整为最大堆。 4. 重复步骤 2 和步骤 3,直到所有元素都排序完成。 以下是堆排序的 C++ 代码实现: ```cpp #include <iostream> using namespace std; // 调整以 root 为根的子树为最大堆 void heapify(int arr[], int n, int root) { int largest = root; // 假设根节点最大 int left = 2 * root + 1; // 左子节点索引 int right = 2 * root + 2; // 右子节点索引 // 若左子节点大于根节点,更新最大值索引 if (left < n && arr[left] > arr[largest]) { largest = left; } // 若右子节点大于最大值节点,更新最大值索引 if (right < n && arr[right] > arr[largest]) { largest = right; } // 若最大值不是根节点,交换根节点和最大值 if (largest != root) { swap(arr[root], arr[largest]); // 递归调整交换后的子树 heapify(arr, n, largest); } } void heapSort(int arr[], int n) { // 构建最大堆 for (int i = n / 2 - 1; i >= 0; i--) { heapify(arr, n, i); } // 逐步取出最大值,调整堆 for (int i = n - 1; i > 0; i--) { swap(arr[0], arr[i]); heapify(arr, i, 0); } } int main() { int arr[] = {4, 10, 3, 5, 1}; int n = sizeof(arr) / sizeof(arr[0]); heapSort(arr, n); cout << "Sorted array: "; for (int i = 0; i < n; i++) { cout << arr[i] << " "; } cout << endl; return 0; } ``` 以上就是堆排序的基本思路和实现方法。堆排序的时间复杂度为 O(nlogn),其中 n 为数组的长度。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值