HDU 1498(50 years, 50 colors)x轴和y轴的二分匹配

这道题没什么,就是枚举每一个点,枚举完的标记一下,然后直接进行二分匹配。刚刚开始你可能会有疑问x,y轴怎么进行二分匹配呢?

首先,就抽取其中的一行x'来看,则这一行里面会有(x',y1),(x',y2),(x',y3)......(x',yn)的点是可以选择的,也就是说x'可以和yi组成一个匹配,然后每行都是如此,那么这么一来就转化为了二分匹配的问题(即当你选了一行的某一个点后,其它的点就不可以碰)。

此题是一行一样颜色的气球直接就会被插爆,问你那些颜色会在k times不会被插爆。

#include<stdio.h>
#include<string.h>
#include<algorithm>
#define N 100+5
using namespace std;
int map[N][N];
int n,k;
int used[55],vis[N],ans[55],link[N];
void Init()
{
	memset(used,0,sizeof(used));
}
bool cmp(int a,int b)
{
	return a<b;
}
bool dfs(int x,int color)
{
	for(int i=1;i<=n;i++)//y轴
	{
		if(!vis[i]&&map[x][i]==color)
		{
			vis[i]=1;
			if(link[i]==-1||dfs(link[i],color))
			{
				link[i]=x;
				return true;
			}
		}
	}
	return false;
}
int main()
{
	while(~scanf("%d%d",&n,&k),n+k)
	{
		int sum,p=0,mark=0;
		Init();
		for(int i=1;i<=n;i++)
		{
			for(int j=1;j<=n;j++)
			{
				scanf("%d",&map[i][j]);
			} 
		}
		for(int i=1;i<=n;i++)//开始枚举
	    {
	    	for(int j=1;j<=n;j++)
	    	{
	    		memset(link,-1,sizeof(link));
	    		if(!used[map[i][j]])//如果这个颜色没有被访问过,就继续
	    		{
	    			used[map[i][j]]=1;//标记访问
	    			sum=0;
	    			for(int u=1;u<=n;u++)//x轴
	    			{
	    				memset(vis,0,sizeof(vis));
	    				if(dfs(u,map[i][j]))
	    				{
	    					sum++;
	    				}
	    			}
	    			if(sum>k)如果它的最大匹配数比k大,则此气球不会被插爆(请不要觉得猥琐)
				    {
					    mark=1;
					    ans[p++]=map[i][j];//存入数组中
				    }
	    		}
	    	}
	    }
	    if(!mark) 
	    {
	    	printf("-1\n");
	    	continue;
	    }
		sort(ans,ans+p,cmp);//按要求排序输出
		int falg=0;
		for(int i=0;i<p;i++)
		{
			if(falg==0)
			{
				falg=1;
				printf("%d",ans[i]);
			}
			else
			{
				printf(" %d",ans[i]);
			}
		}
		printf("\n");
	}
	return 0;
} 

你可能感兴趣的:(二分匹配,匈牙利算法)