poj1719 - Shooting Contest

                                  想看更多的解题报告: http://blog.csdn.net/wangjian8006/article/details/7870410
                                  转载请注明出处:http://blog.csdn.net/wangjian8006

题目大意:给你一个r*c的矩阵,然后这个矩阵每一列有两个格子是白色的,然后你在每列上开一枪,可以打这列的任意的白色格子,问你打到的格子所在的行数是不是包括所有行了,如果包括所有行输出你在1-c列打抢的格子的行,当然这可以是不唯一的,

如果打不到就输出NO
题目首先给出有几组数据,对于每组数据第一行代表r和c,接下来c行有两个数,代表这行的哪两列是白色格子

 

解题思路:和以前那个经典的二分匹配差不多,就是在一行打一枪可以消灭所有行那个题目,很容易想到二分匹配,我们将白色格子的行指向列,然后求最大匹配,用行去匹配列,看是否能够得到的匹配数是r,如果是r的话则代表所有的行都可以被打到,然后每列对应的行,如果匹配数不是r就输出NO
当然有特例,如果r>c,这种情况开c抢根本不可能达到r行,所以直接输出NO

 

/*
Memory 1172K
Time    47MS
*/
#include <iostream>
using namespace std;
#define MAXV 1010

int r,c;
bool map[MAXV][MAXV],use[MAXV];
int vx[MAXV],vy[MAXV];

bool dfs(int x){
	int i;
	for(i=1;i<=c;i++){
		if(map[x][i] && !use[i]){
			use[i]=1;
			if(vy[i]==-1 || dfs(vy[i])){
				vy[i]=x;
				vx[x]=i;
				return true;
			}
		}
	}
	return false;
}

int hungary(){
	int i,num=0;
	memset(vx,-1,sizeof(vx));
	memset(vy,-1,sizeof(vy));
	for(i=1;i<=r;i++){
		if(vx[i]==-1){
			memset(use,0,sizeof(use));
			if(dfs(i)) num++;
		}
	}
	return num;
}

int main(){
	int i,a,b,Case,j;
	scanf("%d",&Case);
	while(Case--){
		scanf("%d%d",&r,&c);
		memset(map,0,sizeof(map));
		for(i=1;i<=c;i++){
			scanf("%d%d",&a,&b);
			map[a][i]=1;
			map[b][i]=1;
		}
		if(c<r || hungary()!=r){
			printf("NO\n");
		}else{
			for(j = 1;j <= c;j++)
				if(vy[j] != -1)
					 printf("%d ",vy[j]);
				 else
					 for(i=1;i<=r;i++)
						 if(map[i][j]){
							 printf("%d ",i);
							 break;
						}
			printf("\n");
		}
	}
	return 0;
}

你可能感兴趣的:(c)