题意:给定一个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 - 二分图匹配的更多相关文章

  1. UVa 二分图匹配 Examples

    这些都是刘汝佳的算法训练指南上的例题,基本包括了常见的几种二分图匹配的算法. 二分图是这样一个图,顶点分成两个不相交的集合X , Y中,其中同一个集合中没有边,所有的边关联在两个集合中. 给定一个二分 ...

  2. 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 ...

  3. uva 12083 Guardian of Decency (二分图匹配)

    uva 12083 Guardian of Decency Description Frank N. Stein is a very conservative high-school teacher. ...

  4. UVA 1663 Purifying Machine (二分图匹配,最大流)

    题意: 给m个长度为n的模板串,模板串由0和1和*三种组成,且每串至多1个*,代表可0可1.模板串至多匹配2个串,即*号改成0和1,如果没有*号则只能匹配自己.问:模板串可以缩减为几个,同样可以匹配原 ...

  5. UVA 11045-My T-shirt suits me(二分图匹配)

    题意:有N件T恤,N是6的倍数,因为有6种型号,每种件数相同,有M个人,每个人有两种型号的T恤适合他,每个人可以挑其中的一种,问能否所有的人都能分配到T恤. 解析:典型的二分图匹配,每N/6为同种T恤 ...

  6. uva 12549

    12549 - Sentry Robots Time limit: 1.000 seconds We need to guard a set of points of interest using s ...

  7. POJ 1274 裸二分图匹配

    题意:每头奶牛都只愿意在她们喜欢的那些牛栏中产奶,告诉每头奶牛愿意产奶的牛棚编号,求出最多能分配到的牛栏的数量. 分析:直接二分图匹配: #include<stdio.h> #includ ...

  8. BZOJ1433 ZJOI2009 假期的宿舍 二分图匹配

    1433: [ZJOI2009]假期的宿舍 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 2375  Solved: 1005[Submit][Sta ...

  9. HDU1281-棋盘游戏-二分图匹配

    先跑一个二分图匹配,然后一一删去匹配上的边,看能不能达到最大匹配数,不能这条边就是重要边 /*----------------------------------------------------- ...

随机推荐

  1. effective java —— 终结方法守卫者

    目录: effective java —— 终结方法守卫者 effective java 第2章:创建和销毁对象.第7条 : 避免使用终结方法.最后的“终结方法守卫者 (finalizer guard ...

  2. 使用Autolayout对多行文本Label进行布局,高度不准确的解决办法!

    BUG描述: 今天公司的项目中发现了一个BUG,大概给大家描述一下,tabbleView有一个tableFooterView,这个footView中有一个Label,是多行显示文本,程序用的是Auto ...

  3. PHP 数据安全问题总结

    总结:关键的判断,比较尽量使用=== 类型和值都比较的恒等比较 1.if($var) $var 遵循boolean 转换. 当转换为 boolean 时,以下值被认为是 FALSE: 布尔值 FALS ...

  4. ExtJs 之 ComboBox级联使用

    刚接触ExtJs不到一周,项目使用ExtJs框架,有个版块用到了combobox的级联(两级),遇到了一系列的问题,两天来一直查API.网络资料,终于解决了. 先列出遇到的一系列问题(也许你也遇到过! ...

  5. chattr和lsattr

    这两个命令是和权限有关 1.chattr +i carlton.txt 对carlton.txt文件进行锁定,谁也不能进行任何修改,取消的话可以chattr -i carlton.txt 就可以 2. ...

  6. 远程线程DLL注入64位进程

    int main() { BOOL bFlag = FALSE; char *szDllName = "MSGDLL.dll"; //bFlag = EnablePrivilege ...

  7. Java中获取长度length和size的问题

    1.length属性是针对Java中的数组来说的,要求数组的长度可以用其length属性: 2.length()方法是针对字符串来说的,要求一个字符串的长度就要用到它的length()方法: 3.ja ...

  8. Makefile 一点一滴(三)—— 尝试简单的变量替换

    上一版的 makefile: TestCpp : ./debug/TestCpp.o g++ -o TestCpp ./debug/TestCpp.o ./debug/TestCpp.o : ./sr ...

  9. 灰常好的开源项目[c/c++]

    ClibPDF http://cosoft.net.cn http://www2s.biglobe.ne.jp/~Nori/ruby/dist/ClibPDF-ALPHA-20010519.tar.g ...

  10. SET-UID程序漏洞实验

    一.实验描述 Set-UID 是Unix系统中的一个重要的安全机制.当一个Set-UID程序运行的时候,它被假设为具有拥有者的权限.例如,如果程序的拥有者是root,那么任何人运行这个程序时都会获得程 ...