1. 程式人生 > >【BZOJ4950】lydsy七月月賽 C 二分圖最大匹配

【BZOJ4950】lydsy七月月賽 C 二分圖最大匹配

for 但是 需要 com 成了 strong div mic printf

【BZOJ4950】lydsy七月月賽 C

題面

題解:比較直接的想法就是:每行,每列的最大值都留下,剩下的格子都變成1。但是如果一個格子既是行的最大值又是列的最大值,那麽我們只需要把它留下即可。這就變成了一個二分圖最大匹配問題,亂搞即可。

#include <cstdio>
#include <cstring>
#include <iostream>
using namespace std;
typedef long long ll;
int n,m,cnt,now;
ll ans;
ll map[110][110],s1[110],s2[110];
int to[10010],next[10010],head[110],vis[110],from[110];
void add(int a,int b)
{
	to[++cnt]=b,next[cnt]=head[a],head[a]=cnt;
}
int dfs(int x)
{
	for(int i=head[x];i;i=next[i])
	{
		if(vis[to[i]])	continue;
		vis[to[i]]=1;
		if(!from[to[i]]||dfs(from[to[i]]))
		{
			from[to[i]]=x;
			return 1;
		}
	}
	return 0;
}
int main()
{
	scanf("%d%d",&n,&m);
	int i,j;
	for(i=1;i<=n;i++)	for(j=1;j<=m;j++)
		scanf("%lld",&map[i][j]),s1[i]=max(s1[i],map[i][j]),s2[j]=max(s2[j],map[i][j]),ans+=map[i][j];
	for(i=1;i<=n;i++)	for(j=1;j<=m;j++)
	{
		if(map[i][j])
		{
			ans--;
			if(s1[i]>1&&s1[i]==s2[j])	add(i,j);
		}
	}
	for(i=1;i<=n;i++)	if(s1[i])	ans-=s1[i]-1;
	for(i=1;i<=m;i++)	if(s2[i])	ans-=s2[i]-1;
	for(i=1;i<=n;i++)	if(s1[i])	memset(vis,0,sizeof(vis)),ans+=dfs(i)*(s1[i]-1);
	printf("%lld",ans);
	return 0;
}

【BZOJ4950】lydsy七月月賽 C 二分圖最大匹配