UVA 12549 - 二分图匹配
题意:给定一个Y行X列的网格,网格种有重要位置和障碍物。要求用最少的机器人看守所有重要的位置,每个机器人放在一个格子里,面朝上下左右四个方向之一发出激光直到射到障碍物为止,沿途都是看守范围。机器人不会阻挡射线。
“#”表示障碍物,“*”表示重要的位置,箭头表示最终机器人匹配的位置,求出机器人能够匹配出的最少位置个数。
分析:首先看题解是二分图匹配,但是建图目前还没想到呢。每个机器人绝对都在重要位置上,假设每个重要位置上都有机器人,可以保护哪几个重要物品。看是否能够匹配得来
#include<cstdio>
#include<memory.h>
#include <iostream>
using namespace std;
#define repu(i, a, b) for(int i = (a); i < (b); i++)
#define MAX 202
bool flag,visit[MAX]; ///记录V2中的某个点是否被搜索过
int match[MAX]; ///记录与V2中的点匹配的点的编号
int cow, stall; ///二分图中左边、右边集合中顶点的数目
int head[MAX];
struct edge
{
int to,next;
} e[];
int index,X,Y;
void addedge(int u,int v)
{
///向图中加边的算法,注意加上的是有向边
///u为v的后续节点既是v---->u
e[index].to=v;
e[index].next=head[u];
head[u]=index;
index++;
}
/// 匈牙利(邻接表)算法
bool dfs(int u)
{
int i,v;
for(i = head[u]; i != ; i = e[i].next)
{
v = e[i].to;
if(!visit[v]) ///如果节点v与u相邻并且未被查找过
{
visit[v] = true; ///标记v为已查找过
if(match[v] == - || dfs(match[v])) ///如果i未在前一个匹配M中,或者i在匹配M中,但是从与i相邻的节点出发可以有增广路径
{
match[v] = u; ///记录查找成功记录,更新匹配M(即“取反”)
return true; ///返回查找成功
}
}
}
return false;
}
int g[MAX][MAX];
pair<int, int> Map[MAX][MAX];
int n,m;
void MaxMatch()
{
int i,sum=;
memset(match,-,sizeof(match));
for(i = ; i < X; ++i)
{
memset(visit,false,sizeof(visit));///清空上次搜索时的标记
if(dfs(i)) ///从节点i尝试扩展
sum++;
}
printf("%d\n",sum);
}
void build()
{
int r = -, c = -;
repu(i,,n+)
{
bool flag = true;
repu(j,,m+)
{
if(g[i][j] == )
{
///如果没有障碍物,说明一个机器人就可以解决,否则就得加一个机器人
if(flag)
++r;
Map[i][j].first = r;
flag = false;
}
if(g[i][j] == )
flag = true;
}
}
repu(j,,m+)
{
bool flag = true;
repu(i,,n+)
{
if(g[i][j] == )
{
if(flag) ++c;
Map[i][j].second = c;
flag = false;
}
if(g[i][j] == ) flag = true;
}
}
X = r + ;
repu(i,,n+)
repu(j,,m+)
if(g[i][j] == )
{
addedge(Map[i][j].first,Map[i][j].second);
cout<<Map[i][j].first<<"--"<<Map[i][j].second<<endl;
}
}
void init()
{
int a, x, y;
memset(g, , sizeof(g));
scanf("%d%d%d", &n, &m, &a);
while(a--)
{
scanf("%d%d", &x, &y);
g[x][y] = ;
}
scanf("%d", &a);
while(a--)
{
scanf("%d%d", &x, &y);
g[x][y] = ;
}
}
///和POJ 3041的区别就是有障碍物
///因为有障碍物,所以需要进行行列拆分。。。即build
int main()
{
int T,a,b,x,y,l,r;
scanf("%d",&T);
while(T--)
{
memset(head,,sizeof(head)); ///切记要初始化
index = ;
init();
build();
MaxMatch();
}
return ;
}
UVA 12549 - 二分图匹配的更多相关文章
- UVa 二分图匹配 Examples
这些都是刘汝佳的算法训练指南上的例题,基本包括了常见的几种二分图匹配的算法. 二分图是这样一个图,顶点分成两个不相交的集合X , Y中,其中同一个集合中没有边,所有的边关联在两个集合中. 给定一个二分 ...
- POJ 2289 Jamie's Contact Groups / UVA 1345 Jamie's Contact Groups / ZOJ 2399 Jamie's Contact Groups / HDU 1699 Jamie's Contact Groups / SCU 1996 Jamie's Contact Groups (二分,二分图匹配)
POJ 2289 Jamie's Contact Groups / UVA 1345 Jamie's Contact Groups / ZOJ 2399 Jamie's Contact Groups ...
- uva 12083 Guardian of Decency (二分图匹配)
uva 12083 Guardian of Decency Description Frank N. Stein is a very conservative high-school teacher. ...
- UVA 1663 Purifying Machine (二分图匹配,最大流)
题意: 给m个长度为n的模板串,模板串由0和1和*三种组成,且每串至多1个*,代表可0可1.模板串至多匹配2个串,即*号改成0和1,如果没有*号则只能匹配自己.问:模板串可以缩减为几个,同样可以匹配原 ...
- UVA 11045-My T-shirt suits me(二分图匹配)
题意:有N件T恤,N是6的倍数,因为有6种型号,每种件数相同,有M个人,每个人有两种型号的T恤适合他,每个人可以挑其中的一种,问能否所有的人都能分配到T恤. 解析:典型的二分图匹配,每N/6为同种T恤 ...
- uva 12549
12549 - Sentry Robots Time limit: 1.000 seconds We need to guard a set of points of interest using s ...
- POJ 1274 裸二分图匹配
题意:每头奶牛都只愿意在她们喜欢的那些牛栏中产奶,告诉每头奶牛愿意产奶的牛棚编号,求出最多能分配到的牛栏的数量. 分析:直接二分图匹配: #include<stdio.h> #includ ...
- BZOJ1433 ZJOI2009 假期的宿舍 二分图匹配
1433: [ZJOI2009]假期的宿舍 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 2375 Solved: 1005[Submit][Sta ...
- HDU1281-棋盘游戏-二分图匹配
先跑一个二分图匹配,然后一一删去匹配上的边,看能不能达到最大匹配数,不能这条边就是重要边 /*----------------------------------------------------- ...
随机推荐
- 深入springMVC源码------文件上传源码解析(下篇)
在上篇<深入springMVC------文件上传源码解析(上篇) >中,介绍了springmvc文件上传相关.那么本篇呢,将进一步介绍springmvc 上传文件的效率问题. 相信大部分 ...
- ftgl 绘制文字
FTFont* ftfont = new FTGLPixmapFont(); ftfont->Open("D:/SIMHEI.ttf"); ftfont->FaceSi ...
- CentOS_7.2服务器前期
一.禁用SELinux:# 永久禁用,需要重启生效: sed -i 's/SELINUX=enforcing/SELINUX=disabled/g' /etc/sysconfig/selinux se ...
- 非对称加密算法--DH
注意:本节内容主要参考自<Java加密与解密的艺术(第2版)>第8章“高等加密算法--非对称加密算法” 11.1.非对称加密算法 特点: 发送方和接收方均有一个密钥对(公钥+私钥),其中公 ...
- 14073102(CCDIKRecoil)
[目标] CCDIKRecoil [思路] 1 CCDIK和Recoil的结合 2 Recoil的回弹机制,逐渐回到原来位置 3 添加一个Recoil基类 [步骤] 1 将\Src\GameFrame ...
- Codeforces Round #211 (Div. 2) D题(二分,贪心)解题报告
---恢复内容开始--- 题目地址 简要题意: n个小伙子一起去买自行车,他们有每个人都带了一些钱,并且有公有的一笔梦想启动资金,可以分配给任何小伙子任何数值,当然分配权在我们的手中.现在给出m辆自行 ...
- Android中View的基础知识
View的界限 View就是我们看到的界面,有四个界限范围分别是, Top/Left, Bottom/Right,坐标系从左上到右下.这四个值可以通过任何View的子类调用getTop()/get.. ...
- SQLite常用函数
length(column_name) 取得相应栏位的长度 substr(column_name, start, length) 截取某个栏位相应长度的值
- 1、jvm的体系结构
jvm包括两子系统两组件 a.两子系统:Class Loader子系统,Execution engine子系统 b.两组件:Runtime Date Area 和 Native Interface
- 曲线拟合的最小二乘法(基于OpenCV实现)
1.原理 在现实中经常遇到这样的问题,一个函数并不是以某个数学表达式的形式给出,而是以一些自变量与因变量的对应表给出,老师讲课的时候举的个例子是犯罪人的身高和留下的脚印长,可以测出一些人的数据然后得到 ...