最大匹配也叫最大边独立集,就是无向图中能取出两两不相邻的边的最大集合。

二分图最大匹配可以用最大流来解。

如果题目没有墙,那就是一道经典的二分图最大匹配问题:

把地图上的行和列分别作为点的X部和Y部,地图上每一块空地看作边,边的两个端点就是它所在的x行y列。这样,求最大边独立集即可。

而这一题有墙,然后我不会了。。

其实这题的建模也是一样的,也是行和列作为点,空地作为边:

  • 对于每一行把被墙分隔的每一块连通的区域缩成一点,列也一样;
  • 行缩成的点作为X部,列Y部;
  • 某行连通区域最多就只能在区域内某一块空地放机器人,列也是一样;
  • 如果某行连通区域和某列连通区域相交,就连边。
  1. #include<cstdio>
  2. #include<cstring>
  3. #include<queue>
  4. #include<algorithm>
  5. using namespace std;
  6. #define INF (1<<30)
  7. #define MAXN 2555
  8. #define MAXM 555555
  9.  
  10. struct Edge{
  11. int v,cap,flow,next;
  12. }edge[MAXM];
  13. int vs,vt,NE,NV;
  14. int head[MAXN];
  15.  
  16. void addEdge(int u,int v,int cap){
  17. edge[NE].v=v; edge[NE].cap=cap; edge[NE].flow=;
  18. edge[NE].next=head[u]; head[u]=NE++;
  19. edge[NE].v=u; edge[NE].cap=; edge[NE].flow=;
  20. edge[NE].next=head[v]; head[v]=NE++;
  21. }
  22.  
  23. int level[MAXN];
  24. int gap[MAXN];
  25. void bfs(){
  26. memset(level,-,sizeof(level));
  27. memset(gap,,sizeof(gap));
  28. level[vt]=;
  29. gap[level[vt]]++;
  30. queue<int> que;
  31. que.push(vt);
  32. while(!que.empty()){
  33. int u=que.front(); que.pop();
  34. for(int i=head[u]; i!=-; i=edge[i].next){
  35. int v=edge[i].v;
  36. if(level[v]!=-) continue;
  37. level[v]=level[u]+;
  38. gap[level[v]]++;
  39. que.push(v);
  40. }
  41. }
  42. }
  43.  
  44. int pre[MAXN];
  45. int cur[MAXN];
  46. int ISAP(){
  47. bfs();
  48. memset(pre,-,sizeof(pre));
  49. memcpy(cur,head,sizeof(head));
  50. int u=pre[vs]=vs,flow=,aug=INF;
  51. gap[]=NV;
  52. while(level[vs]<NV){
  53. bool flag=false;
  54. for(int &i=cur[u]; i!=-; i=edge[i].next){
  55. int v=edge[i].v;
  56. if(edge[i].cap!=edge[i].flow && level[u]==level[v]+){
  57. flag=true;
  58. pre[v]=u;
  59. u=v;
  60. //aug=(aug==-1?edge[i].cap:min(aug,edge[i].cap));
  61. aug=min(aug,edge[i].cap-edge[i].flow);
  62. if(v==vt){
  63. flow+=aug;
  64. for(u=pre[v]; v!=vs; v=u,u=pre[u]){
  65. edge[cur[u]].flow+=aug;
  66. edge[cur[u]^].flow-=aug;
  67. }
  68. //aug=-1;
  69. aug=INF;
  70. }
  71. break;
  72. }
  73. }
  74. if(flag) continue;
  75. int minlevel=NV;
  76. for(int i=head[u]; i!=-; i=edge[i].next){
  77. int v=edge[i].v;
  78. if(edge[i].cap!=edge[i].flow && level[v]<minlevel){
  79. minlevel=level[v];
  80. cur[u]=i;
  81. }
  82. }
  83. if(--gap[level[u]]==) break;
  84. level[u]=minlevel+;
  85. gap[level[u]]++;
  86. u=pre[u];
  87. }
  88. return flow;
  89. }
  90.  
  91. char map[][];
  92. int main(){
  93. int t,n,m;
  94. scanf("%d",&t);
  95. for(int cse=; cse<=t; ++cse){
  96. scanf("%d%d",&n,&m);
  97. for(int i=; i<n; ++i){
  98. for(int j=; j<m; ++j) scanf(" %c",&map[i][j]);
  99. }
  100.  
  101. int d0[][]={},d1[][]={},cnt=;
  102. int rown=,coln=;
  103. for(int i=; i<n; ++i){
  104. for(int j=; j<m; ){
  105. if(map[i][j]=='o'){
  106. ++rown;
  107. while(j<m && map[i][j]!='#') d0[i][j]=rown,++j;
  108. }else ++j;
  109. }
  110. }
  111. for(int j=; j<m; ++j){
  112. for(int i=; i<n; ){
  113. if(map[i][j]=='o'){
  114. ++coln;
  115. while(i<n && map[i][j]!='#') d1[i][j]=coln+rown,++i;
  116. }else ++i;
  117. }
  118. }
  119.  
  120. vs=; vt=rown+coln+; NV=vt+; NE=;
  121. memset(head,-,sizeof(head));
  122. for(int i=; i<=rown; ++i) addEdge(vs,i,);
  123. for(int i=; i<=coln; ++i) addEdge(i+rown,vt,);
  124. for(int i=; i<n; ++i){
  125. for(int j=; j<m; ++j){
  126. if(map[i][j]=='o' && d0[i][j] && d1[i][j]) addEdge(d0[i][j],d1[i][j],);
  127. }
  128. }
  129.  
  130. printf("Case :%d\n",cse);
  131. printf("%d\n",ISAP());
  132. }
  133. return ;
  134. }

ZOJ1654 Place the Robots(二分图最大匹配)的更多相关文章

  1. zoj1654 Place the Robots 二分图最大匹配

    题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=654 将每一行的包含空地的区域编号 再将每一列的包含空地的区域编号 然 ...

  2. POJ 2226二分图最大匹配

    匈牙利算法是由匈牙利数学家Edmonds于1965年提出,因而得名.匈牙利算法是基于Hall定理中充分性证明的思想,它是二部图匹配最常见的算法,该算法的核心就是寻找增广路径,它是一种用增广路径求二分图 ...

  3. POJ2239 Selecting Courses(二分图最大匹配)

    题目链接 N节课,每节课在一个星期中的某一节,求最多能选几节课 好吧,想了半天没想出来,最后看了题解是二分图最大匹配,好弱 建图: 每节课 与 时间有一条边 #include <iostream ...

  4. poj 2239 二分图最大匹配,基础题

    1.poj 2239   Selecting Courses   二分图最大匹配问题 2.总结:看到一个题解,直接用三维数组做的,很巧妙,很暴力.. 题意:N种课,给出时间,每种课在星期几的第几节课上 ...

  5. UESTC 919 SOUND OF DESTINY --二分图最大匹配+匈牙利算法

    二分图最大匹配的匈牙利算法模板题. 由题目易知,需求二分图的最大匹配数,采取匈牙利算法,并采用邻接表来存储边,用邻接矩阵会超时,因为邻接表复杂度O(nm),而邻接矩阵最坏情况下复杂度可达O(n^3). ...

  6. 二分图最大匹配的K&#246;nig定理及其证明

     二分图最大匹配的K?nig定理及其证明 本文将是这一系列里最短的一篇,因为我只打算把K?nig定理证了,其它的废话一概没有.    以下五个问题我可能会在以后的文章里说,如果你现在很想知道的话,网上 ...

  7. POJ3057 Evacuation(二分图最大匹配)

    人作X部:把门按时间拆点,作Y部:如果某人能在某个时间到达某门则连边.就是个二分图最大匹配. 时间可以二分枚举,或者直接从1枚举时间然后加新边在原来的基础上进行增广. 谨记:时间是个不可忽视的维度. ...

  8. HDU:过山车(二分图最大匹配)

    http://acm.hdu.edu.cn/showproblem.php?pid=2063 题意:有m个男,n个女,和 k 条边,求有多少对男女可以搭配. 思路:裸的二分图最大匹配,匈牙利算法. 枚 ...

  9. UOJ #78 二分图最大匹配

    #78. 二分图最大匹配 从前一个和谐的班级,有 nl 个是男生,有 nr 个是女生.编号分别为 1,…,nl 和 1,…,nr. 有若干个这样的条件:第 v 个男生和第 u 个女生愿意结为配偶. 请 ...

随机推荐

  1. xcode注释

    新开的项目需要先开发iOS版本,所以又把好久没写的iOS捡起来了,之前都是手动注释,最近是越来越懒了,所以在网上找了一个自动注释的插件,啊哈,其实有时候还真的挺怀念用Eclipse的时候,不过不用羡慕 ...

  2. delphi 枚举类型

    枚举类型定义了一系列有序值的集合.枚举变量就是从这个既定的集合中取某个值.集合中的有序值可以称为元素,元素一般从0开始索引(也就是元素的顺序号). 定义一个枚举类型,采用以下的格式: type typ ...

  3. Linux Apache 怎么修改工作模式

    Apache默认为prefork模式,主要是考虑到稳定性的原因. 要切换到worker模式,则需要登录到linux上,进行如下操作: 进入/usr/sbin目录 cd /usr/sbin 将当前的pr ...

  4. Android自定义Dialog

    Android开发过程中,常常会遇到一些需求场景——在界面上弹出一个弹框,对用户进行提醒并让用户进行某些选择性的操作, 如退出登录时的弹窗,让用户选择“退出”还是“取消”等操作. Android系统提 ...

  5. C/C++语法知识:typedef struct 用法详解

    第一篇:typedef struct与struct的区别 1. 基本解释 typedef为C语言的关键字,作用是为一种数据类型定义一个新名字.这里的数据类型包括内部数据类型(int,char等)和自定 ...

  6. Redis使用介绍

    Redis 是一个高性能的key-value数据库. redis的出现,很大程度补偿了memcached这类keyvalue存储的不足,在部 分场合可以对关系数据库起到很好的补充作用.它提供了Pyth ...

  7. 简单制作 OS X Yosemite 10.10 正式版U盘USB启动安装盘方法教程

    转自: http://www.iplaysoft.com/osx-yosemite-usb-install-drive.html http://jingyan.baidu.com/article/02 ...

  8. 开发Web Service的几种方式

    本文作者在学习使用Java开发Web Service(不包括Restful)时,由于不知道Java有这么多框架支持开发Web Service一度陷入迷惘,不知道这些框架各有 什么不同,各有什么优缺点. ...

  9. 【leetcode】Letter Combinations of a Phone Number

    Letter Combinations of a Phone Number Given a digit string, return all possible letter combinations ...

  10. 转 MYSQL学习(一)

    第一期主要是学习MYSQL的基本语法,陆续还有第二.第三.第四期,大家敬请期待o(∩_∩)o 语法的差异 我这里主要说语法的不同 1.默认约束 区别:mysql里面DEFAULT关键字后面是不用加括号 ...