专栏文章

题解:P3386 【模板】二分图最大匹配

P3386题解参与者 1已保存评论 0

文章操作

快速查看文章及其快照的属性,并进行相关操作。

当前评论
0 条
当前快照
1 份
快照标识符
@mipb25gj
此快照首次捕获于
2025/12/03 09:05
3 个月前
此快照最后确认于
2025/12/03 09:05
3 个月前
查看原文

匈牙利算法

我们用一个二维数组来存图,然后遍历 nn 个节点,看看这 nn 个点中,有多少个可以连。

核心代码

CPP
bool dfs(int u) {
	for (int i = 1; i <= m; i++) {
		if (f[u][i] && !vis[i]) {
			vis[i] = 1;
			if (!match[i] || dfs(match[i])) {
				match[i] = u;
				return 1;
			}
		}
	}
	return 0;
}
遍历 mm 条边,看看那些边跟 uu 联系上了,接着看当前节点有没有被连上,如果没有,更新访问数组。接下来看当前节点的匹配节点是什么,如果没有,那么把匹配节点设成 uu;如果有,尝试把它连得点断掉,把当前节点匹配上 uu

代码

CPP
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<vector>
#include<cmath>

using namespace std;

const int N = 1e3;
bool f[N][N];
bool vis[N];
int match[N], n, m, e; 

bool dfs(int u) {
	for (int i = 1; i <= m; i++) {
		if (f[u][i] && !vis[i]) {
			vis[i] = 1;
			if (!match[i] || dfs(match[i])) {
				match[i] = u;
				return 1;
			}
		}
	}
	return 0;
}

int main() {
	scanf("%d%d%d", &n, &m, &e);
	for (int i = 1; i <= e; i++) {
		int x, y;
		scanf("%d%d", &x, &y);
		f[x][y] = 1;
	}
	int ans = 0;
	for (int i = 1; i <= n; i++) {
		memset(vis, 0, sizeof(vis));
		if (dfs(i)) ans++;
	}
	printf("%d\n", ans);
	return 0;
}

评论

0 条评论,欢迎与作者交流。

正在加载评论...