这道题没什么,就是枚举每一个点,枚举完的标记一下,然后直接进行二分匹配。刚刚开始你可能会有疑问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;
}