题意:给定一个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. occ添加新的捕捉模式

    Load (theSelection, theShape, theType, theDeflection, theDeviationAngle, isAutoTriangulation, thePri ...

  2. Python笔记总结week4

      1. Built-in functions 函数可能遇到的问题:下面例子函数改变了函数需要传入的参数 li = [11,22,33,44] def f1(arg): arg.append(55) ...

  3. Oracle PL/SQL

    PL/SQL 简介 PL/SQL 是过程语言(Procedural Language)与结构化查询语言(SQL)结合而成的编程语言,是对 SQL 的扩展,它支持多种数据类型,如大对象和集合类型,可使用 ...

  4. 自己不懂的SQL语句用法

    left  join:是SQL语言中的查询类型,即连接查询.它的全称为左外连接,是外连接的一种. 连接通常可以在select语句的from子句或where子句中建立,其语法格式为: select  c ...

  5. iOS开发_MVC设计模式

    MVC,是一种主流的设计模式,本博文总结一下自己对MVC的看法和理解,浅知拙见,难登大雅之堂,如有欠缺遗漏,还望各位指正. MVC是Model-View-Controller 的缩写,Model 是指 ...

  6. PHPExcel读取Excel文件的实现代码

    <?php require_once 'PHPExcel.php'; /**对excel里的日期进行格式转化*/ function GetData($val){ $jd = GregorianT ...

  7. Bootstrap <基础二十八>列表组

    列表组.列表组件用于以列表形式呈现复杂的和自定义的内容.创建一个基本的列表组的步骤如下: 向元素 <ul> 添加 class .list-group. 向 <li> 添加 cl ...

  8. eclipse 字体、背景、自动提示设置

    1 字体设置 点击最上面菜单栏的“Window”---“preferences”弹出属性界面 General---  Appearance---Colors and Fronts,找到Java 选择“ ...

  9. js数组去重的方法

    //数组去重 Array.prototype.unique = function() { var newArr = [], hash = {}; for(var i=0, len=this.lengt ...

  10. C#获取一个目录下的所有文件名

    今天在做图像训练的时候发现需要把一大堆图片进行处理再读进分类器,本来是用C++写的,结果发现并不会,于是就用回了我最爱的C#,结果棒棒哒. 代码如下,简单粗暴,比网上C++的语法好看多了 using ...